SpringBoot 全局异常拦截器

点击上方“Java基基”,选择“设为星标”

做积极的人,而不是积极废人!

每天 14:00 更新文章,每天掉亿点点头发...

源码精品专栏

 

来源:blog.csdn.net/qq_36922927/
article/details/82026683

d40c1efdf2ca7274a2940e243391a883.png


通常jsr303参数校验,由于返回的数据提示很不友好(bindException),

1f54a7eb4fae7c9c55b81622724f4ceb.png

需要定义全局异常拦截器,将信息友好的显示给用户

本文以处理登录为例

定义全局异常拦截器:继承自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)

CodeMsg.java

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条件分支即可):

GlobalException.java

/**
 * 全局异常类
 */
@Data
public class GlobalException extends RuntimeException{
  private CodeMsg codeMsg;
  public GlobalException(CodeMsg codeMsg){
    super(codeMsg.toString());
    this.codeMsg=codeMsg;

  }
}

推荐下自己做的 Spring Boot 的实战项目:

https://github.com/YunaiV/ruoyi-vue-pro

看下效果:

使用异常处理器之前,我处理登陆的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("成功");
    }

推荐下自己做的 Spring Cloud 的实战项目:

https://github.com/YunaiV/onemall

添加异常处理器之后:

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,手机号格式不正确

211938440e877d087d268286ba9229d0.png

2,密码错误

848171d2eb9c7c7417f9f47c6e9de736.png

3,手机号不存在

5a9e24dfea05faac224b8b9f1c449b00.png

4,手机号与密码均正确

dd0059d3f10527589f7f54eaa740ebb3.png

欢迎加入我的知识星球,一起探讨架构,交流源码。加入方式,长按下方二维码噢

631766341fb6ebcd87a8c8a0584c423b.png

已在知识星球更新源码解析如下:

4d759beef871386b53d6fe28919e3bd0.png

daa1668f7ee38bf3affb1911a1119ebf.png

89ee144b2e8e717cc0307fb661b8bcdd.png

0129ffd234262deb9e01444ae153d6db.png

最近更新《芋道 SpringBoot 2.X 入门》系列,已经 101 余篇,覆盖了 MyBatis、Redis、MongoDB、ES、分库分表、读写分离、SpringMVC、Webflux、权限、WebSocket、Dubbo、RabbitMQ、RocketMQ、Kafka、性能测试等等内容。

提供近 3W 行代码的 SpringBoot 示例,以及超 6W 行代码的电商微服务项目。

获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。

文章有帮助的话,在看,转发吧。
谢谢支持哟 (*^__^*)
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值