【216期】面试官问:Spring Boot 全局异常拦截器?
点击上方“Java精选”,选择“设为星标”
别问别人为什么,多问自己凭什么!
下方有惊喜,留言必回,有问必答!
每天 08:15 更新文章,每天进步一点点...
通常jsr303参数校验,由于返回的数据提示很不友好(bindException),
需要定义全局异常拦截器,将信息友好的显示给用户
本文以处理登录为例
定义全局异常拦截器:继承自RuntimeException
GlobalExceptionHandler.java
import org.springframework.validation.BindException;
@ControllerAdvice
@ResponseBody
public class GlobalExceptionHandler {
@ExceptionHandler(value = Exception.class)
public Result<String> exceptionHandler(HttpServletRequest request, Exception e){
//绑定异常是需要明确提示给用户的
if(e instanceof BindException){
BindException exception=(BindException) e;
List<ObjectError> errors=exception.getAllErrors();
String msg=errors.get(0).getDefaultMessage();//获取自错误信息
return Result.error(CodeMsg.SERVER_BIND_ERROR.fillArgs(msg));
//将具体错误信息设置到CodeMsg中返回
}
// 其余异常简单返回为服务器异常
return Result.error(CodeMsg.SERVER_ERROR);
}
}
由于之前的CodeMsg类,只接收code,msg参数构造CodeMsg对象,如果需要定制ErrorException的codeMsg,
需要接收一个异常内容的参数:
只需要添加一个生成异常CodeMsg对象的方法:CodeMsg fillArgs(Object ... args)
public class CodeMsg {
private int code;
private String msg;
//通用异常
public static CodeMsg SUCCESS = new CodeMsg(0, "success");
public static CodeMsg SERVER_ERROR = new CodeMsg(500100, "服务端异常");
//注意 %s ,格式化字符串
public static CodeMsg SERVER_BIND_ERROR = new CodeMsg(500101, "服务端绑定异常:%s");
//登录模块 5002XX
public static CodeMsg MSG_PASSWORD_IS_EMPTY = new CodeMsg(500201, "密码不能为空!");
public static CodeMsg MSG_MOBILE_ERROR = new CodeMsg(500202, "手机号格式不正确!");
public static CodeMsg MSG_MOBILE_IS_EMPTY = new CodeMsg(500203, "手机号不能为空!");
public static CodeMsg MSG_MOBILE_NOT_EXIST = new CodeMsg(500204, "手机号不存在!");
public static CodeMsg MSG_PASSWORD_ERROR = new CodeMsg(500205, "密码错误!");
//商品模块 5003XX
//订单模块 5004XX
//秒杀模块 5005XX
private CodeMsg(int code, String msg) {
this.code = code;
this.msg = msg;
}
/**
*@created 23:03 2018/8/24
*@author wangwei
*@params
*@return 异常CodeMsg 对象生成方法
*/
public CodeMsg fillArgs(Object ... args){
int code=this.code;
String message=String.format(msg,args);
return new CodeMsg(code,message);
}
public int getCode() {
return code;
}
public String getMsg() {
return msg;
}
}
在业务代码中专注处理业务,而不是返回各种CodeMsg(比如这里只需要知道登录时成功还是失败,其余情况直接抛出异常),可以直接抛出异常,添加一个全局异常类,根据CodeMsg来生成异常, 交由GlobalExceptionHandler全局异常处理器处理(在其中增加if条件分支即可):
/**
* 全局异常类
*/
@Data
public class GlobalException extends RuntimeException{
private CodeMsg codeMsg;
public GlobalException(CodeMsg codeMsg){
super(codeMsg.toString());
this.codeMsg=codeMsg;
}
}
看下效果:
使用异常处理器之前,我处理登陆的service方法代码是这样的:
public CodeMsg login(LoginVal loginVal){
if(null==loginVal){
throw new GlobalException(CodeMsg.SERVER_ERROR);
}
String mobile=loginVal.getMobile();
String password=loginVal.getPassword();
MiaoshaUser user=miaoShaUserDao.getUserById(Long.parseLong(mobile));
if(null==user){
return CodeMsg.MSG_MOBILE_NOT_EXIST;
}
//
if(!user.getPassword().equals(MD5Util.formPassword2DbPass(password,user.getSalt())) ){
return CodeMsg.MSG_PASSWORD_ERROR;
}
return CodeMsg.SUCCESS;
}
除此之外,controller方法:还有业务逻辑
public Result doLogin(@Valid LoginVal loginVal){
System.out.println("doLogin");
log.info(loginVal);
CodeMsg loginCodeMsg=userService.login(loginVal);
if(loginCodeMsg.getCode()!=0){
return Result.error(loginCodeMsg);
}
return Result.success("成功");
}
添加异常处理器之后:
service的处理login的业务代码是这样的:
//登录的记过只想知道是true还是false,其余均是抛出全局异常,交由异常处理器处理
public boolean login(LoginVal loginVal){
if(null==loginVal){
throw new GlobalException(CodeMsg.SERVER_ERROR);
}
String mobile=loginVal.getMobile();
String password=loginVal.getPassword();
MiaoshaUser user=miaoShaUserDao.getUserById(Long.parseLong(mobile));
if(null==user){
throw new GlobalException( CodeMsg.MSG_MOBILE_NOT_EXIST);
}
if(!user.getPassword().equals(MD5Util.formPassword2DbPass(password,user.getSalt())) ){
throw new GlobalException(CodeMsg.MSG_PASSWORD_ERROR);
}
return true;
}
controller方法是这样的:无业务逻辑
由于各种null,以及密码不正确等问题都在service抛出GlobalException,这里自然只能得到true
public Result<Boolean> doLogin(@Valid LoginVal loginVal){
System.out.println("doLogin");
log.info(loginVal);
userService.login(loginVal);
//由于各种null,以及密码不正确等问题都在service抛出GlobalException,这里自然只能得到true
return Result.success(true);
}
修改全局异常处理器
GlobalExceptionHandler.java是这样的:
@ControllerAdvice
@ResponseBody
public class GlobalExceptionHandler {
@ExceptionHandler(value = Exception.class)
public Result<String> exceptionHandler(HttpServletRequest request, Exception e){
//全局异常处理逻辑
if(e instanceof GlobalException){
return Result.error(((GlobalException) e).getCodeMsg());
}
//绑定异常处理逻辑
else if(e instanceof BindException){
BindException exception=(BindException) e;
List<ObjectError> errors=exception.getAllErrors();
String msg=errors.get(0).getDefaultMessage();
return Result.error(CodeMsg.SERVER_BIND_ERROR.fillArgs(msg));
}
return Result.error(CodeMsg.SERVER_ERROR);
}
}
看添加上异常处理器之后页面效果:
存在的手机号是:12345678901,密码是123456
1,手机号格式不正确
2,密码错误
3,手机号不存在
版权声明:本文为CSDN博主「大卫不加班」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
https://blog.csdn.net/qq_36922927/article/details/82026683
3000+ 道面试题在线刷,最新、最全 Java 面试题!
【210期】Spring Boot 多线程环境下,解决多个定时器冲突问题
【212期】 架构师经常提起领用 DNS 解析实现负载均衡是个什么“鬼”?
最近有很多人问,有没有读者&异性交流群,你懂的!想知道如何加入。加入方式很简单,有兴趣的同学,只需要点击下方卡片,回复“加群”,即可免费加入交流群!