创建一个数据库,这里以yytryproject为例
CREATE TABLE `edu_teacher` (
`id` char(19) NOT NULL COMMENT '讲师ID',
`name` varchar(20) NOT NULL COMMENT '讲师姓名',
`intro` varchar(500) NOT NULL DEFAULT '' COMMENT '讲师简介',
`career` varchar(500) DEFAULT NULL COMMENT '讲师资历,一句话说明讲师',
`level` int(10) unsigned NOT NULL COMMENT '头衔 1高级讲师 2首席讲师',
`avatar` varchar(255) DEFAULT NULL COMMENT '讲师头像',
`sort` int(10) unsigned NOT NULL DEFAULT '0' COMMENT '排序',
`is_deleted` tinyint(1) unsigned NOT NULL DEFAULT '0' COMMENT '逻辑删除 1(true)已删除, 0(false)未删除',
`gmt_create` datetime NOT NULL COMMENT '创建时间',
`gmt_modified` datetime NOT NULL COMMENT '更新时间',
PRIMARY KEY (`id`),
UNIQUE KEY `uk_name` (`name`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='讲师';
往里边随便插入几条数据:
INSERT INTO `yytryproject`.`edu_teacher`(`id`, `name`, `intro`, `career`, `level`, `avatar`, `sort`, `is_deleted`, `gmt_create`, `gmt_modified`) VALUES ('1', '测试人员一号', '普通讲师简介', '普通讲师资历', 1, 'www.baidu.com', 1, 0, '2019-09-03 13:45:00', '2020-09-03 13:45:18');
以下规约只针对本模块,更全面的文档参考《阿里巴巴Java开发手册》:
1、库名与应用名称尽量一致
2、表名、字段名必须使用小写字母或数字,禁止出现数字开头,
3、表名不使用复数名词
4、表的命名最好是加上“业务名称_表的作用”。如,edu_teacher_
5、表必备三字段:id, gmt_create, gmt_modified
说明:
其中 id 必为主键,类型为 bigint unsigned、单表时自增、步长为 1。
(如果使用分库分表集群部署,则id类型为verchar,非自增,业务中使用分布式id生成器)
gmt_create, gmt_modified 的类型均为 datetime 类型,前者现在时表示主动创建,后者过去分词表示被
动更新。
6、单表行数超过 500 万行或者单表容量超过 2GB,才推荐进行分库分表。 说明:如果预计三年后的数
据量根本达不到这个级别,请不要在创建表时就分库分表。
guli_edu.sql 1
7、表达是与否概念的字段,必须使用 is_xxx 的方式命名,数据类型是 unsigned tinyint (1 表示是,0 表
示否)。
说明:任何字段如果为非负数,必须是 unsigned。
注意:POJO 类中的任何布尔类型的变量,都不要加 is 前缀。数据库表示是与否的值,使用 tinyint 类
型,坚持 is_xxx 的 命名方式是为了明确其取值含义与取值范围。
正例:表达逻辑删除的字段名 is_deleted,1 表示删除,0 表示未删除。
8、小数类型为 decimal,禁止使用 float 和 double。 说明:float 和 double 在存储的时候,存在精度损
失的问题,很可能在值的比较时,得到不 正确的结果。如果存储的数据范围超过 decimal 的范围,建议
将数据拆成整数和小数分开存储。
9、如果存储的字符串长度几乎相等,使用 char 定长字符串类型。
10、varchar 是可变长字符串,不预先分配存储空间,长度不要超过 5000,如果存储长度大于此值,定
义字段类型为 text,独立出来一张表,用主键来对应,避免影响其它字段索 引效率。
11、唯一索引名为 uk_字段名;普通索引名则为 idx_字段名。
说明:uk_ 即 unique key;idx_ 即 index 的简称
12、不得使用外键与级联,一切外键概念必须在应用层解决。外键与级联更新适用于单机低并发,不适
合分布式、高并发集群;级联更新是强阻塞,存在数据库更新风暴的风险;外键影响数据库的插入速度。
guli-parent:在线教学根目录(父工程),管理四个子模块: canal-client:canal数据库表同步模块(统计同步数据) common:公共模块父节点
common-util:工具类模块,所有模块都可以依赖于它
service-base:service服务的base包,包含service服务的公共配置类,所有service模块依赖于它spring-security:认证与授权模块,需要认证授权的service服务依赖于它
infrastructure:基础服务模块父节点
api-gateway:api网关服务
service:api接口服务父节点
service-acl:用户权限管理api接口服务(用户管理、角色管理和权限管理等) service-cms:cms api接口服务
service-edu:教学相关api接口服务service-msm:短信api接口服务service-order:订单相关api接口服务service-oss:阿里云oss api接口服务
service-statistics:统计报表api接口服务service-ucenter:会员api接口服务service-vod:视频点播api接口服务
删除该文件下边的src文件
节点后面添加 pom 类型
删除pom.xml中的
内容
org.springframework.boot
spring-boot-starter
org.springframework.boot
spring-boot-starter-test
test
添加
确定依赖的版本
对这个项目进行 版本控制
1.8
0.0.1-SNAPSHOT
3.0.5
2.0
2.7.0
3.1.0
2.10.1
3.17
1.3.1
2.6
4.5.1
0.7.0
4.3.3
3.1.0
2.15.2
1.4.11
1.4.11
1.2.28
2.8.2
20170516
1.7
1.1.0
zx
0.2.2.RELEASE
5.5.8
配置
锁定依赖的版本
这里不会直接导入,只 会后面项目引入后才导入
org.springframework.cloud
spring-cloud-dependencies
Hoxton.RELEASE
pom
import
org.springframework.cloud
spring-cloud-alibaba-dependencies
${cloud-alibaba.version}
pom
import
com.baomidou
mybatis-plus-boot-starter
${mybatis-plus.version}
org.apache.velocity
velocity-engine-core
${velocity.version}
io.springfox
springfox-swagger2
${swagger.version}
io.springfox
springfox-swagger-ui
${swagger.version}
com.aliyun.oss
aliyun-sdk-oss
${aliyun.oss.version}
joda-time
joda-time
${jodatime.version}
org.apache.poi
poi
${poi.version}
org.apache.poi
poi-ooxml
${poi.version}
commons-fileupload
commons-fileupload
${commons-fileupload.version}
commons-io
commons-io
${commons-io.version}
org.apache.httpcomponents
httpclient
${httpclient.version}
com.google.code.gson
gson
${gson.version}
io.jsonwebtoken
jjwt
${jwt.version}
com.aliyun
aliyun-java-sdk-core
${aliyun-java-sdk-core.version}
com.aliyun.oss
aliyun-sdk-oss
${aliyun-sdk-oss.version}
com.aliyun
aliyun-java-sdk-vod
${aliyun-java-sdk-vod.version}
com.aliyun
aliyun-java-vod-upload
${aliyun-java-vod-upload.version}
com.aliyun
aliyun-sdk-vod-upload
${aliyun-sdk-vod-upload.version}
com.alibaba
fastjson
${fastjson.version}
org.json
json
${json.version}
commons-dbutils
commons-dbutils
${commons-dbutils.version}
com.alibaba.otter
canal.client
${canal.client.version}
p6spy
p6spy
3.9.1
cn.hutool
hutool-all
${hutool-version}
注意:该下的
aliyun-java-vod-upload aliyun-sdk-vod-upload
2个依赖无法下载下来,因为不是开源,需要自己手动下载
链接:https://pan.baidu.com/s/1zRu3tA0zWL3WArHe0UIB6Q
提取码:idea
下载之后进入到对应的jar包文件中,输入cmd
执行代码:
mvn install:install-file -DgroupId=com.aliyun -DartifactId=aliyun-sdk-vod-upload -Dversion=1.4.12 -Dpackaging=jar -Dfile=aliyun-java-vod-upload-1.4.11.jar
aliyun-java-vod-upload 执行代码:
mvn install:install-file -DgroupId=com.aliyun -DartifactId=aliyun-java-vod-upload -Dversion=1.4.11 -Dpackaging=jar -Dfile=aliyun-java-vod-upload-1.4.11.jar
官方下载的版本需要更改版本在pom中修改以及执行代码中也需要修改版本,这里以1.4.11为例
org.springframework.boot
spring-boot-starter-web
com.baomidou
mybatis-plus-boot-starter
mysql
mysql-connector-java
org.apache.velocity
velocity-engine-core
io.springfox
springfox-swagger2
io.springfox
springfox-swagger-ui
org.projectlombok
lombok
org.apache.poi
poi
org.apache.poi
poi-ooxml
commons-fileupload
commons-fileupload
org.apache.httpcomponents
httpclient
commons-io
commons-io
com.google.code.gson
gson
io.jsonwebtoken
jjwt
com.aliyun
aliyun-java-sdk-core
com.aliyun.oss
aliyun-sdk-oss
com.aliyun
aliyun-java-sdk-vod
com.aliyun
aliyun-java-vod-upload
com.aliyun
aliyun-sdk-vod-upload
com.alibaba
fastjson
org.json
json
commons-dbutils
commons-dbutils
com.alibaba.otter
canal.client
com.aliyun.oss
aliyun-sdk-oss
joda-time
joda-time
com.yy
service_base
0.0.1-SNAPSHOT
p6spy
p6spy
junit
junit
4.12
前三个依赖先不需要,后边需要放开,否则前边代码跑不起来
#端口号
server:
port: 8001
#服务名
spring:
application:
name: service-edu
#环境配置
profiles:
active: dev
#数据库连接
datasource:
url: jdbc:p6spy:mysql://localhost:3306/yytryproject?serverTimeZone=GMT%2B8
username: root
password: root
# 执行sql分析执行
driver-class-name: com.p6spy.engine.spy.P6SpyDriver
validation-query: SELECT 1
mybatis-plus日志
mybatis-plus:
configuration:
log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
import com.baomidou.mybatisplus.core.exceptions.MybatisPlusException;
import com.baomidou.mybatisplus.core.toolkit.StringPool;
import com.baomidou.mybatisplus.generator.AutoGenerator;
import com.baomidou.mybatisplus.generator.InjectionConfig;
import com.baomidou.mybatisplus.generator.config.*;
import com.baomidou.mybatisplus.generator.config.po.TableInfo;
import com.baomidou.mybatisplus.generator.config.rules.NamingStrategy;
import com.baomidou.mybatisplus.generator.engine.VelocityTemplateEngine;
import org.apache.commons.lang3.StringUtils;
import java.util.ArrayList;
import java.util.List;
import java.util.Scanner;
//代码生成器
public class CodeGenerator {
/**
*
* 读取控制台内容
*
*/
public static String scanner(String tip) {
Scanner scanner = new Scanner(System.in);
StringBuilder help = new StringBuilder();
help.append("请输入" + tip + ":");
System.out.println(help.toString());
if (scanner.hasNext()) {
String ipt = scanner.next();
if (StringUtils.isNotBlank(ipt)) {
return ipt;
}
}
throw new MybatisPlusException("请输入正确的" + tip + "!");
}
public static void main(String[] args) {
// 代码生成器
AutoGenerator mpg = new AutoGenerator();
// 全局配置
GlobalConfig gc = new GlobalConfig();
String projectPath = System.getProperty("user.dir");
gc.setOutputDir(projectPath + "/service/service_edu/src/main/java");
gc.setAuthor("yy");
gc.setOpen(false);
// 设置名字
gc.setControllerName("%sController");
gc.setServiceName("%sService");
gc.setServiceImplName("%sServiceImpl");
gc.setMapperName("%sDao");
gc.setXmlName("%sDao");
// 设置 resultMap
gc.setBaseResultMap(true);
gc.setBaseColumnList(true);
// gc.setFileOverride(true);
gc.setSwagger2(true); //实体属性 Swagger2 注解
mpg.setGlobalConfig(gc);
// 数据源配置
DataSourceConfig dsc = new DataSourceConfig();
dsc.setUrl("jdbc:mysql://localhost:3306/yytryproject?serverTimezone=UTC&serverTimeZone=GMT%2B8");
dsc.setDriverName("com.mysql.cj.jdbc.Driver");
dsc.setUsername("root");
dsc.setPassword("root");
mpg.setDataSource(dsc);
// 自定义配置
InjectionConfig cfg = new InjectionConfig() {
@Override
public void initMap() {
// to do nothing
}
};
// 包配置
PackageConfig pc = new PackageConfig();
// pc.setModuleName(scanner("模块名"));
pc.setParent("com.yy");
pc.setMapper("dao");
mpg.setPackageInfo(pc);
// 如果模板引擎是 velocity
String templatePath = "/templates/mapper.xml.vm";
// 自定义输出配置
List focList = new ArrayList<>();
// 自定义配置会被优先输出
focList.add(new FileOutConfig(templatePath) {
@Override
public String outputFile(TableInfo tableInfo) {
// 自定义输出文件名 , 如果你 Entity 设置了前后缀、此处注意 xml 的名称会跟着发生变化!!
return projectPath+"/service/service_edu/src/main/resources/mapper/"
+ "/" + tableInfo.getEntityName() + "Dao" + StringPool.DOT_XML;
}
});
cfg.setFileOutConfigList(focList);
mpg.setCfg(cfg);
// 配置模板
TemplateConfig templateConfig = new TemplateConfig();
templateConfig.setXml(null);
mpg.setTemplate(templateConfig);
// 策略配置
StrategyConfig strategy = new StrategyConfig();
strategy.setNaming(NamingStrategy.underline_to_camel);
strategy.setColumnNaming(NamingStrategy.underline_to_camel);
strategy.setEntityLombokModel(true);
strategy.setRestControllerStyle(true);
// 写于父类中的公共字段
// strategy.setSuperEntityColumns("id");
strategy.setInclude(scanner("表名,多个英文逗号分割").split(","));
strategy.setControllerMappingHyphenStyle(true);
strategy.setTablePrefix(pc.getModuleName() + "_");
mpg.setStrategy(strategy);
mpg.setTemplateEngine(new VelocityTemplateEngine());
mpg.execute();
}
}
@RestController
@RequestMapping("/eduService/teacher")
public class EduTeacherController {
@Autowired
private EduTeacherService eduTeacherService;
//查询讲师表所有数据
@GetMapping("/findAll")
public List list(){
return eduTeacherService.list(null);
}
}
在edu包下创建config包,创建MyBatisPlusConfig.java
@Configuration
@EnableTransactionManagement
@MapperScan("com.yy.service.mapper")
public class MyBatisPlusConfig {
}
在MyBatisPlusConfig下配置
/**
* SQL 执行性能分析插件
* 开发环境使用,线上不推荐。 maxTime 指的是 sql 最大执行时长
*/
@Bean
@Profile({"dev","test"})// 设置 dev test 环境开启
public PerformanceInterceptor performanceInterceptor() {
PerformanceInterceptor performanceInterceptor = new PerformanceInterceptor();
performanceInterceptor.setMaxTime(1000);//ms,超过此处设置的ms则sql不执行
performanceInterceptor.setFormat(true);
return performanceInterceptor;
}
创建启动类 EduApplication.java,注意启动类的创建位置
直接在里边输入main回车即可
启动配置类;在网页上输入http://localhost:8001/eduService/teacher/findAll
//逻辑删除讲师
@DeleteMapping("{id}")
public boolean deleteTeacherById(@PathVariable String id){
return eduTeacherService.removeById(id);
}
MyBatisPlusConfig中配置
@Bean
public ISqlInjector sqlInjector() {
return new LogicSqlInjector();
}
@TableLogic
private Boolean isDeleted;
浏览器从一个域名的网页去请求另一个域名的资源时,域名、端口、协议任一不同,都是跨域 。前后端分离开发中,需要考虑ajax跨域的问题。
这里我们可以从服务端解决这个问题
在Controller类上添加注解@CrossOrigin
前后端分离开发模式中,api文档是最好的沟通方式。
Swagger 是一个规范和完整的框架,用于生成、描述、调用和可视化 RESTful 风格的 Web 服务。
生成在线接口文档
方便接口测试
- 及时性 (接口变更后,能够及时准确地通知相关前后端开发人员)
- 规范性 (并且保证接口的规范性,如接口的地址,请求方式,参数及响应格式和错误信息)
- 一致性 (接口信息一致,不会出现因开发人员拿到的文档版本不一致,而出现分歧)
- 可测性 (直接在接口文档上进行测试,以方便理解业务)
1)、创建common公共模块,为了让所有模块都可使用
org.springframework.boot
spring-boot-starter-web
provided
com.baomidou
mybatis-plus-boot-starter
provided
org.projectlombok
lombok
provided
io.springfox
springfox-swagger2
provided
io.springfox
springfox-swagger-ui
provided
org.springframework.boot
spring-boot-starter-data-redis
创建包 com.achang.yy.swagger ,创建类 SwaggerConfig.java
package com.yy.swagger;
import com.google.common.base.Predicates;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import springfox.documentation.builders.ApiInfoBuilder;
import springfox.documentation.builders.PathSelectors;
import springfox.documentation.service.ApiInfo;
import springfox.documentation.service.Contact;
import springfox.documentation.spi.DocumentationType;
import springfox.documentation.spring.web.plugins.Docket;
import springfox.documentation.swagger2.annotations.EnableSwagger2;
@Configuration
@EnableSwagger2
public class SwaggerConfig {
@Bean
public Docket webApiConfig(){
return new Docket(DocumentationType.SWAGGER_2)
.groupName("webApi")
.apiInfo(webApiInfo())
.select()
.paths(Predicates.not(PathSelectors.regex("/admin/.*")))
.paths(Predicates.not(PathSelectors.regex("/error.*")))
.build();
}
private ApiInfo webApiInfo(){
return new ApiInfoBuilder()
.title("网站-课程中心API文档")
.description("本文档描述了课程中心微服务接口定义")
.version("1.0")
.contact(new Contact("yy", "http://yy.com",
"[email protected]"))
.build();
}
}
service/pom.xml中引入service-base的坐标
service-base
com.achang
0.0.1-SNAPSHOT
6)、在service-edu启动类上添加注解
@ComponentScan(basePackages = "com.yy")
来扫描SwaggerConfig这个类
@SpringBootApplication
@ComponentScan(basePackages = {"com.yy"})
public class EduApplication {
public static void main(String[] args) {
SpringApplication.run(EduApplication.class, args);
}
}
访问:http://localhost:8001/swagger-ui.html
可以添加一些自定义设置,例如:
一般情况下,统一返回数据格式没有固定的格式,只要能描述清楚返回的数据状态以及要返回的具体数据就可以。
但是一般会包含状态码、返回消息、数据这几部分内容
我们的定义统一结果:
{
"success": 布尔, //响应是否成功
"code": 数字, //响应码
"message": 字符串, //返回消息
"data": HashMap //返回数据,放在键值对中
}
创建包com.achang.commonutils,创建接口 ResultCode.java,抽取状态码,便于后期修改
package com.achang.commonutils;
public interface ResultCode {
//状态码:成功
public static Integer SUCCESS = 20000;
//状态码:失败
public static Integer ERROR = 20001;
}
package com.yy.commonUtils;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
import java.util.HashMap;
import java.util.Map;
@Data
public class R {
@ApiModelProperty("是否成功")
private boolean success;
@ApiModelProperty("响应码")
private Integer code;
@ApiModelProperty("返回信息")
private String message;
@ApiModelProperty("返回数据")
private Map data = new HashMap();
public R() {
}
//成功静态方法
public static R ok(){
R r = new R();
r.setCode(ResultCode.SUCCESS);
r.setMessage("成功");
r.setSuccess(true);
return r;
}
//成功静态方法
public static R error(){
R r = new R();
r.setCode(ResultCode.ERROR);
r.setMessage("失败");
r.setSuccess(false);
return r;
}
public R success(Boolean success){
this.setSuccess(success);
return this;
}
public R code(Integer code){
this.setCode(code);
return this;
}
public R message(String message){
this.setMessage(message);
return this;
}
public R data(String key,Object value){
this.data.put(key,value);
return this;
}
public R data(Map map){
this.setData(map);
return this;
}
}
1)、在service模块中添加依赖
com.achang
common-utils
0.0.1-SNAPSHOT
2)、修改Controller中的返回结果
修改返回结果为R
/**
* 分页插件
*/
@Bean
public PaginationInterceptor paginationInterceptor() {
return new PaginationInterceptor();
}
//3.分页查询讲师功能
@ApiOperation("分页查询讲师功能")
@GetMapping("pageTeacher/{current}/{limit}")
public R pageTeacher(@PathVariable Long current,@PathVariable Long limit){
Page pageTeacher = new Page<>(limit,current);
IPage teacherPageList = eduTeacherService.page(pageTeacher, null);
List records = teacherPageList.getRecords();
long total = teacherPageList.getTotal();
Map data = new HashMap();
data.put("total",total);
data.put("rows",records);
return R.ok().data(data);
}
根据讲师名称name,讲师头衔level、讲师入驻时间gmt_create(时间段)查询
创建service_edu中com.achang.eduservice.entity.vo包,创建TeacherQuery.java查询对象
@Data
public class TeacherQuery {
private Long current;
private Long limit;
@ApiModelProperty(value = "教师名称,模糊查询")
private String name;
@ApiModelProperty(value = "头衔 1普通讲师 2高级讲师 3超级讲师")
private Integer level;
@ApiModelProperty(value = "查询开始时间", example = "2019-01-01 10:10:10")
private String begin;//注意,这里使用的是String类型,前端传过来的数据无需进行类型转换
// private Date begin;
@ApiModelProperty(value = "查询结束时间", example = "2019-12-01 10:10:10")
private String end;
// private Date end;
}
TeacherQuery的属性根据前端需要的查询条件来设置
@RequestBody(required = false)表示增加参数TeacherQuery teacherQuery,非必选
//4.条件查询带分页方法
@ApiOperation("条件查询带分页方法")
@PostMapping("pageTeacherCondition")
public R pageTeacherCondition(@RequestBody(required = false) TeacherQuery teacherQuery){
Long current = teacherQuery.getCurrent();
Long limit = teacherQuery.getLimit();
IPage teacherPageList = eduTeacherService.pageList(current,limit,teacherQuery);
List records = teacherPageList.getRecords();
long total = teacherPageList.getTotal();
Map data = new HashMap<>();
data.put("tatol", total);
data.put("rows", records);
return R.ok().data(data);
}
2)、service
public interface EduTeacherService extends IService {
IPage pageList(Long current, Long limit, TeacherQuery teacherQuery);
}
2)、实现类impl
@Service
public class EduTeacherServiceImpl extends ServiceImpl implements EduTeacherService {
public IPage pageList(Long current, Long limit, TeacherQuery teacherQuery) {
Integer level = teacherQuery.getLevel();
String name = teacherQuery.getName();
String begin = teacherQuery.getBegin();
String end = teacherQuery.getEnd();
LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper<>();
queryWrapper.eq(level!=null, EduTeacher::getLevel,level)
.like(StringUtils.isNotBlank(name),EduTeacher::getName,name)
.ge(begin!=null,EduTeacher::getGmtCreate,begin)
.le(end!=null, EduTeacher::getGmtModified,end);
return this.page(new Page<>(current,limit), queryWrapper);
}
}
给新增数据和修改数据,添加新建时间和修改时间的自动填充封装功能
创建包myMetaObjectHandler,创建自动填充类 MyMetaObjectHandler
@Component
public class MyMetaObjectHandler implements MetaObjectHandler{
public void insertFill(MetaObject metaObject) {
this.setFieldValByName("gmtCreate",new Date(),metaObject);
this.setFieldValByName("gmtModified",new Date(),metaObject);
}
public void updateFill(MetaObject metaObject) {
this.setFieldValByName("gmtModified",new Date(),metaObject);
}
}
//5.新增教师
@ApiOperation("新增教师接口")
@PostMapping("addTeacher")
public R addTeacher(@RequestBody EduTeacher eduTeacher){
boolean save = eduTeacherService.save(eduTeacher);
return save?R.ok():R.error().message("添加失败请联系管理员");
}
//6.1新增教师
@ApiOperation("根据id查询教师")
@GetMapping("getById/{id}")
public R getById(@PathVariable String id){
EduTeacher teacherInfo = eduTeacherService.getById(id);
return teacherInfo!=null?R.ok().data("data", teacherInfo):R.error().message("教师不存在");
}
@ApiOperation("根据id修改教师资料")
@PostMapping("updateById")
public R updateById(@RequestBody EduTeacher eduTeacher){
EduTeacher teacherInfo = eduTeacherService.getById(eduTeacher.getId());
if (teacherInfo==null){
return R.error().message("教师不存在");
}
boolean updateStatus = eduTeacherService.updateById(eduTeacher);
return updateStatus?R.ok():R.error().message("修改失败");
}
引入依赖:因为要使用common-utils中的返回类R
在service_base中创建统一异常处理类exceptionHandler/GlobalExceptionHandler.java:
@ControllerAdvice
public class GlobalExceptionHandler {
//全局异常处理
@ExceptionHandler(Exception.class)
//因为他不在Controller中。没有@RestController,所以数据不会返回,需要加@ResponeseBody返回数据
@ResponseBody
public R error(Exception e){
e.printStackTrace();
return R.error().message("执行了全局异常");
}
}
2)、测试
exceptionHandler/GlobalExceptionHandler.java中添加
//指定异常会执行这个方法
@ExceptionHandler(ArithmeticException.class) //优先执行指定异常,如果没有指定异常则抛出全局异常
//因为他不在Controller中。没有@RestController,所以数据不会返回,需要加@ResponeseBody返回数据
@ResponseBody
public R error(ArithmeticException e){
e.printStackTrace();
return R.error().message("执行了指定异常");
}
2)、测试
在需要用的地方,直接抛出自定义的异常类
exceptionHandler/GlobalExceptionHandler中添加处理这个异常的处理器
//自定义异常会执行这个方法
@ExceptionHandler(RRException.class) //系统不会处理,需要自己手动输出
//因为他不在Controller中。没有@RestController,所以数据不会返回,需要加@ResponeseBody返回数据
@ResponseBody
public R error(RRException e){
e.printStackTrace();
return R.error().code(e.getCode()).message(e.getMsg());
}
4)、测试