网易首页 > 网易号 > 正文 申请入驻

【065期】面试官问:如何实现 Spring Boot 捕获全局异常与参数校验?

0
分享至

>>号外:关注“Java精选”公众号,回复“面试资料”,免费领取资料!“”小程序,3000+ 道面试题在线刷,最新、最全 Java 面试题!

一、为什么要用全局异常处理?

在日常开发中,为了不抛出异常堆栈信息给前端页面,每次编写Controller层代码都要尽可能的catch住所有service层、dao层等异常,代码耦合性较高,且不美观,不利于后期维护。

为解决该问题,计划将Controller层异常信息统一封装处理,且能区分对待Controller层方法返回给前端的String、Map、JSONObject、ModelAndView等结果类型。

二、应用场景是什么?

  • 非常方便的去掉了try catch这类冗杂难看的代码,有利于代码的整洁和优雅

  • 自定义参数校验时候全局异常处理会捕获异常,将该异常统一返回给前端,省略很多if else代码

  • 当后端出现异常时,需要返回给前端一个友好的界面的时候就需要全局异常处理

  • 因为异常时层层向上抛出的,为了避免控制台打印一长串异常信息

三、如何进行全局异常捕获和处理?

一共有两种方法:

  • Spring的AOP(较复杂)

  • @ControllerAdvice结合@ExceptionHandler(简单)

四、@ControllerAdvice和@ExceptionHandler怎么用?

1、Controller Advice字面上意思是“控制器通知”,Advice除了“劝告”、“意见”之外,还有“通知”的意思。

可以将@ExceptionHandler(标识异常类型对应的处理方法)标记的方法提取出来,放到一个类里,并将加上@ControllerAdvice,这样,所有的控制器都可以用了

@ControllerAdvice
public class ControllerHandlers(){
@ExceptionHandler
public String errorHandler(Exception e){
return "error";

2、因为@ControllerAdvice@Componen标记,所以他可以被组件扫描到并放入Spring容器

3、如果只想对一部分控制器通知,比如某个包下边的控制器,就可以这样写:

@ControllerAdvice("com.labor")
public class ControllerHandlers(){}

也可以直接写类名

@ControllerAdvice(basePackageClasses = ***.class)
public class ControllerHandlers(){}

也可以传多个类

@ControllerAdvice(assignableTypes = {***.class,***.class})
public class ControllerHandlers(){}

4、 控制器通知还有一个兄弟,@RestControllerAdvice,如果用了它,错误处理方法的返回值不会表示用的哪个视图,而是会作为HTTP body处理,即相当于错误处理方法加了@ResponseBody注解。,内涵3000+道Java面试题,随时在线刷题。

@RestControllerAdvice
public class ControllerHandlers(){
@ExceptionHandler
public String errorHandler(Exception e){
return "error";

5、@ExceptionHandler注解的方法只能返回一种类型,在前后端分离开发中我们通常返回,统一返回类型和优化错误的提示,我们可以封装我们自己的返回Mappublic class AjaxResult extends HashMap {

private static final long serialVersionUID = 1L;

public static final String CODE_TAG = "code";

public static final String MSG_TAG = "msg";

public static final String DATA_TAG = "data";

/**
* 状态类型
*/
public enum Type {
/**
* 成功
*/
SUCCESS(1),
/**
* 警告
*/
WARN(2),
/**
* 错误
*/
ERROR(0),
/**无权限*/
UNAUTH(3),
/**未登录、登录超时*/
UNLOGIN(4);
private final int value;

Type(int value) {
this.value = value;
}

public int value() {
return this.value;
}
}

/**
* 状态类型
*/
private Type type;

/**
* 状态码
*/
private int code;

/**
* 返回内容
*/
private String msg;

/**
* 数据对象
*/
private Object data;

/**
* 初始化一个新创建的 AjaxResult 对象,使其表示一个空消息。
*/
public AjaxResult() {
}

/**
* 初始化一个新创建的 AjaxResult 对象
* @param type 状态类型
* @param msg 返回内容
*/
public AjaxResult(Type type, String msg) {
super.put(CODE_TAG, type.value);
super.put(MSG_TAG, msg);
}

/**
* 初始化一个新创建的 AjaxResult 对象
* @param type 状态类型
* @param msg 返回内容
* @param data 数据对象
*/
public AjaxResult(Type type, String msg, Object data) {
super.put(CODE_TAG, type.value);
super.put(MSG_TAG, msg);
/* 数据为空的时候,还是需要把参数传给前台 huangqr @2019.7.19
if (StringUtils.isNotNull(data)) {
super.put(DATA_TAG, data);
}*/
super.put(DATA_TAG, data);
}

/**
* 返回成功消息
* @return 成功消息
*/
public static AjaxResult success() {
return AjaxResult.success("操作成功");
}

/**
* 返回成功数据
* @return 成功消息
*/
public static AjaxResult success(Object data) {
return AjaxResult.success("操作成功", data);
}

/**
* 返回成功消息
* @param msg 返回内容
* @return 成功消息
*/
public static AjaxResult success(String msg) {
return AjaxResult.success(msg, null);
}

/**
* 返回成功消息
* @param msg 返回内容
* @param data 数据对象
* @return 成功消息
*/
public static AjaxResult success(String msg, Object data) {
return new AjaxResult(Type.SUCCESS, msg, data);
}

/**
* 返回警告消息
* @param msg 返回内容
* @return 警告消息
*/
public static AjaxResult warn(String msg) {
return AjaxResult.warn(msg, null);
}

/**
* 返回警告消息
* @param msg 返回内容
* @param data 数据对象
* @return 警告消息
*/
public static AjaxResult warn(String msg, Object data) {
return new AjaxResult(Type.WARN, msg, data);
}

/**
* 返回错误消息
* @return
*/
public static AjaxResult error() {
return AjaxResult.error("操作失败");
}

/**
* 返回错误消息
* @param msg 返回内容
* @return 警告消息
*/
public static AjaxResult error(String msg) {
return AjaxResult.error(msg, null);
}

/**
* 返回错误消息
* @param msg 返回内容
* @param data 数据对象
* @return 警告消息
*/
public static AjaxResult error(String msg, Object data) {
return new AjaxResult(Type.ERROR, msg, data);
}

/**
* 无权限返回
* @return
*/
public static AjaxResult unauth() {
return new AjaxResult(Type.UNAUTH, "您没有访问权限!", null);
}
/**
* 无权限
*
* @param msg
* @return com.wanda.labor.framework.web.domain.AjaxResult
* @exception
*/
public static AjaxResult unauth(String msg) {
return new AjaxResult(Type.UNAUTH, msg, null);
}
/**
* 未登录或登录超时。请重新登录
*
* @param
* @return com.wanda.labor.framework.web.domain.AjaxResult
* @exception
*/
public static AjaxResult unlogin() {
return new AjaxResult(Type.UNLOGIN, "未登录或登录超时。请重新登录!", null);
}

public Type getType() {
return type;
}

public void setType(Type type) {
this.type = type;
}

public int getCode() {
return code;
}

public void setCode(int code) {
this.code = code;
}

public String getMsg() {
return msg;
}

public void setMsg(String msg) {
this.msg = msg;
}

public Object getData() {
return data;
}

public void setData(Object data) {
this.data = data;
}

public static class SUCCESS{

public static AjaxResult data(Object data){
return new AjaxResult(Type.SUCCESS, "操作成功 Operation Successful", data);
}

public static AjaxResult iMessagesg(String msg){
return new AjaxResult(Type.SUCCESS, msg, null);
}

public static AjaxResult imsgAndData(String msg,Object data){
return new AjaxResult(Type.SUCCESS, msg, data);
}
}

@Override
public String toString() {
return new ToStringBuilder(this, ToStringStyle.MULTI_LINE_STYLE).append("code", getCode())
.append("msg", getMsg()).append("data", getData()).toString();
}
}
6、完善全局异常处理器@RestControllerAdvice
public class GlobalExceptionHandler {
private static final Logger log = LoggerFactory.getLogger(GlobalExceptionHandler.class);

/**
* 权限校验失败 如果请求为ajax返回json,普通请求跳转页面
*/
@ExceptionHandler(AuthorizationException.class)
public Object handleAuthorizationException(HttpServletRequest request, AuthorizationException e) {
//log.error(e.getMessage(), e);
if (ServletUtils.isAjaxRequest(request)) {
return AjaxResult.unauth(PermissionUtils.getMsg(e.getMessage()));
} else {
ModelAndView modelAndView = new ModelAndView();
modelAndView.setViewName("error/unauth");
return modelAndView;
}
}

/**
* 请求方式不支持
*/
@ExceptionHandler({HttpRequestMethodNotSupportedException.class})
public AjaxResult handleException(HttpRequestMethodNotSupportedException e) {
log.error(e.getMessage(), e);
return AjaxResult.error("不支持' " + e.getMethod() + "'请求");
}

/**
* 拦截未知的运行时异常
*/
@ExceptionHandler(RuntimeException.class)
public AjaxResult notFount(RuntimeException e) {
log.error("运行时异常:", e);
return AjaxResult.error("运行时异常:" + e.getMessage());
}

/**
* 系统异常
*/
@ExceptionHandler(Exception.class)
public AjaxResult handleException(Exception e) {
log.error(e.getMessage(), e);
return AjaxResult.error("服务器错误,请联系管理员");
}

/**
* 校验异常
*/
@ExceptionHandler(value = MethodArgumentNotValidException.class)
public AjaxResult exceptionHandler(MethodArgumentNotValidException e) {
BindingResult bindingResult = e.getBindingResult();
String errorMesssage = "";
for (FieldError fieldError : bindingResult.getFieldErrors()) {
errorMesssage += fieldError.getDefaultMessage() + "!";
}
return AjaxResult.error(errorMesssage);
}

/**
* 校验异常
*/
@ExceptionHandler(value = BindException.class)
public AjaxResult validationExceptionHandler(BindException e) {
BindingResult bindingResult = e.getBindingResult();
String errorMesssage = "";
for (FieldError fieldError : bindingResult.getFieldErrors()) {
errorMesssage += fieldError.getDefaultMessage() + "!";
}
return AjaxResult.error(errorMesssage);
}

/**
* 校验异常
*/
@ExceptionHandler(value = ConstraintViolationException.class)
public AjaxResult ConstraintViolationExceptionHandler(ConstraintViolationException ex) {
Set> constraintViolations = ex.getConstraintViolations();
Iterator> iterator = constraintViolations.iterator();
List msgList = new ArrayList<>();
while (iterator.hasNext()) {
ConstraintViolation cvl = iterator.next();
msgList.add(cvl.getMessageTemplate());
}
return AjaxResult.error(String.join(",",msgList));
}

/**
* 业务异常
*/
@ExceptionHandler(BusinessException.class)
public AjaxResult businessException(BusinessException e) {
log.error(e.getMessage(), e);
return AjaxResult.error(e.getMessage());
}

/**
* 演示模式异常
*/
@ExceptionHandler(DemoModeException.class)
public AjaxResult demoModeException(DemoModeException e) {
return AjaxResult.error("演示模式,不允许操作");
}

}
六、@Validated 校验器注解的异常,也可以一起处理,无需手动判断绑定校验结果 BindingResult/Errors 了

pom文件引入validation的jar包。


org.springframework.bootgroupId>
spring-boot-starter-validationartifactId>
dependency>

等待校验的object

public class Person {
* @PersonName(prefix = "song"):自定义注解
@NotNull
@PersonName(prefix = "song")
private String name;
@Min(value = 18)
@Max(value = 30, message = "超过30岁的不要!")
private Integer age;

自定义注解

  • https://blog.csdn.net/panchang199266/article/details/83050053

使用


* 开启校验注解:@Valid
@RestController
public class PersonController {
@PostMapping("/person")
public Person savePerson(@Valid @RequestBody Person person){
return person;

全局异常处理里有相应的处理方法


* 校验异常
@ExceptionHandler(value = BindException.class)
public AjaxResult validationExceptionHandler(BindException e) {
BindingResult bindingResult = e.getBindingResult();
String errorMesssage = "";
for (FieldError fieldError : bindingResult.getFieldErrors()) {
errorMesssage += fieldError.getDefaultMessage() + "!";
return AjaxResult.error(errorMesssage);

@RequestBody@RequestParam注解的请求实体,校验异常类是不同的

七、自定义异常以及事务回滚public class MyException extends RuntimeException {
//这个地方不要写exception,因为Spring是只对运行时异常进行事务回滚,
//如果抛出的是exception是不会进行事务回滚的。

}

如果是在service层里捕获异常统一去处理,那为了保证事务的回滚,需要抛出RuntimeException

try {
} catch (Exception e) {
e.printStackTrace();
logger.error("发生异常");
throw new RuntimeException();

关于try-catch-finally中,finally的作用,finally设计之初就是为了关闭资源,如果在finally中使用return语句,会覆盖try或者catch的返回值,最常见的就是覆盖异常,即便catch往上抛了异常,也会被覆盖,返回finally中return语句的返回值。,内涵3000+道Java面试题,点击即可进入,随时在线刷题。

作者:Songgp1024 blog.csdn.net/songguopeng/article/details/98961787

往期精选 点击标题可

点个赞,就知道你“在看”!

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
三上悠亚回忆首部作品,解密作为女优出道的目的。

三上悠亚回忆首部作品,解密作为女优出道的目的。

楚门记
2024-06-18 14:54:33
日本人骂人的“八格牙路”,翻译成中文是什么意思?原来比什么都伤人

日本人骂人的“八格牙路”,翻译成中文是什么意思?原来比什么都伤人

书中自有颜如玉
2024-06-19 07:30:18
“人造数学天才”,一个非常拙劣的谎言,彻头彻尾的炒作

“人造数学天才”,一个非常拙劣的谎言,彻头彻尾的炒作

爆角追踪
2024-06-18 17:25:17
这女孩竟然是陈妍希!!!不过照片中这位男士是谁啊?

这女孩竟然是陈妍希!!!不过照片中这位男士是谁啊?

牛叨小史
2024-06-17 18:25:36
上游新闻探访姜萍老家:其家庭年收入中等,上中专不是缺学费,对口招生上本科几率更大

上游新闻探访姜萍老家:其家庭年收入中等,上中专不是缺学费,对口招生上本科几率更大

上游新闻
2024-06-19 19:12:22
“甘油三酯大户”被揪出?医生:能不吃就不吃,别等不适后再后悔

“甘油三酯大户”被揪出?医生:能不吃就不吃,别等不适后再后悔

铠甲护士
2024-06-19 16:42:09
突发!唐治平到电视台下跪求工作,落泪喊话:说800次那不是我妈

突发!唐治平到电视台下跪求工作,落泪喊话:说800次那不是我妈

小咪侃娱圈
2024-06-20 11:05:37
正式离队,山东鲁能离队2人曝光,重金挖来被放弃,王大雷祝福

正式离队,山东鲁能离队2人曝光,重金挖来被放弃,王大雷祝福

东球弟
2024-06-20 11:07:04
孙子结婚当天,82岁奶奶留千字遗书跳楼,遗言:我就是要诅咒你!

孙子结婚当天,82岁奶奶留千字遗书跳楼,遗言:我就是要诅咒你!

古今历史记
2024-06-20 14:27:51
能活到90岁的老人,基本上在60岁的时候,就已经不做这4件事了!

能活到90岁的老人,基本上在60岁的时候,就已经不做这4件事了!

史说历史
2024-06-19 14:16:07
人过了60岁,这四个人就没必要见面了,这是老祖宗的告诫。

人过了60岁,这四个人就没必要见面了,这是老祖宗的告诫。

娱乐洞察点点
2024-06-19 15:21:52
韩国驻菲使馆和外交部就南海问题发表罔顾事实的错误言论,我驻韩使馆:提出严正抗议!

韩国驻菲使馆和外交部就南海问题发表罔顾事实的错误言论,我驻韩使馆:提出严正抗议!

环球网资讯
2024-06-19 17:12:08
CCTV5直播西班牙vs意大利:西班牙两翼年轻起飞 无惧卫冕冠军

CCTV5直播西班牙vs意大利:西班牙两翼年轻起飞 无惧卫冕冠军

智道足球
2024-06-20 12:27:33
姜萍板书中出现11个错误,月考数学仅83分,其在朋友圈说“丢人”

姜萍板书中出现11个错误,月考数学仅83分,其在朋友圈说“丢人”

东东趣谈
2024-06-18 12:31:45
36岁北京不婚主义女博士,嫁26岁东北农村小伙,8年后,现状如何

36岁北京不婚主义女博士,嫁26岁东北农村小伙,8年后,现状如何

小陆搞笑日常
2024-06-18 23:14:40
那些“被出轨”的女星们,每个都是“人间尤物”,你更喜欢哪位?

那些“被出轨”的女星们,每个都是“人间尤物”,你更喜欢哪位?

兰子记
2024-06-11 22:36:02
为啥女生有性生活后,乳头上的小颗粒会变多,这正常吗?

为啥女生有性生活后,乳头上的小颗粒会变多,这正常吗?

荷兰豆爱健康
2024-06-08 07:55:07
不堪入目,女员工惨变领导泄火工具,受害者发声:俩女同事已遭殃

不堪入目,女员工惨变领导泄火工具,受害者发声:俩女同事已遭殃

三月柳
2024-06-06 16:52:16
南方医科大学事件,处理不好将会是医疗界和教育界的“彭宇案”。

南方医科大学事件,处理不好将会是医疗界和教育界的“彭宇案”。

小李子体育
2024-06-20 10:52:51
电力“最大贪官”落马:家中豪车百辆,现金有近10亿,结局如何?

电力“最大贪官”落马:家中豪车百辆,现金有近10亿,结局如何?

天闻地知
2024-06-18 14:03:29
2024-06-20 15:10:44
Java精选
Java精选
一场永远也演不完的戏
1551文章数 3855关注度
往期回顾 全部

科技要闻

小米SU7流量泼天,富贵却被蔚来接住了

头条要闻

深圳网红学位房每平从14万跌到4万 中介:每月都有成交

头条要闻

深圳网红学位房每平从14万跌到4万 中介:每月都有成交

体育要闻

绿军的真老大,开始备战下赛季了

娱乐要闻

叶舒华参加柯震东生日聚会,五毒俱全

财经要闻

日本银行巨头突然爆雷!

汽车要闻

售价11.79-14.39万元 新一代哈弗H6正式上市

态度原创

艺术
游戏
教育
数码
军事航空

艺术要闻

穿越时空的艺术:《马可·波罗》AI沉浸影片探索人类文明

“Uzi疑似复出”登微博热搜!这次是WE战队

教育要闻

三人连环画共60本,甲给乙3本,乙给丙5本后,三人一样多,原来各

数码要闻

消息称苹果公司暂停Vision Pro 2头显的研发工作

军事要闻

普京再送金正恩轿车 两人轮流当司机

无障碍浏览 进入关怀版