@ControllerAdvice

@ControllerAdvice

全局异常处理

步骤 只需要定义类,添加该注解即可定义

@ControllerAdvice
public class MyGlobalExceptionHandler {
    
    //在这里可以细化异常类。用于全局捕获异常
    //可以定义多个
    @ExceptionHandler(Exception.class)
    public ModelAndView customException(Exception e) {
        ModelAndView mv = new ModelAndView();
        mv.addObject("message", e.getMessage());
        mv.setViewName("myerror");
        return mv;
    }
    
    @ResponseStatus(HttpStatus.UNAUTHORIZED)
    @ExceptionHandler(value = ShiroException.class)
    public Result handler(ShiroException e){
        log.error("运行时异常:-------------{}",e);
        return Result.failure(e.getMessage(),401);
    }


    @ResponseStatus(HttpStatus.BAD_REQUEST)
    @ExceptionHandler(value = RuntimeException.class)
    public Result handler(RuntimeException e){

        log.error("运行时异常:--------------------");
        return Result.failure(e.getMessage());
    }


    @ResponseStatus(HttpStatus.BAD_REQUEST)
    @ExceptionHandler(value = MethodArgumentNotValidException.class)
    public Result handler(MethodArgumentNotValidException e){
        log.error("实体校验异常:--------------------");
        BindingResult bindingResult = e.getBindingResult();
        //获取所有校验异常
        ObjectError objectError = bindingResult.getAllErrors().stream().findFirst().get();

        return Result.failure(objectError.getDefaultMessage());
    }
    
    @ResponseStatus(HttpStatus.BAD_REQUEST)
    @ExceptionHandler(value = IllegalArgumentException.class)
    public Result handler(IllegalArgumentException e){
        log.error("Assert异常:--------------------");
        return Result.failure(e.getMessage());
    }
}

定义不同的方法处理不同的异常,

全局数据绑定

全局数据绑定功能可以用来做一些初始化的数据操作

将一些公共的数据定义在添加了@ControllerAdvice注解的类中。这样在每一个Controller的接口中,就能够访问到这些数据

使用 @ModelAttribute 注解标记该方法的返回数据是一个全局数据,默认情况下,这个全局数据的 key 就是返回的变量名,value 就是方法返回值,当然开发者可以通过 @ModelAttribute 注解的 name 属性去重新指定 key。

步骤如下

@ControllerAdvice
public class MyGlobalExceptionHandler {
    @ModelAttribute(name = "message")
    public Map<String,Object> mydata() {
        HashMap<String, Object> map = new HashMap<>();
        map.put("age", 99);
        map.put("gender", "男");
        return map;
    }
}

要访问上面的数据

@Controller
public class UserController {
 
  public ModelAndView detail(@RequestParam("id") long id, 
        //直接通过 该注解获取                   
       @ModelAttribute("message") String message) {
...
  }
}

全局数据预处理

考虑我有两个实体类,Book 和 Author,分别定义如下:

public class Book {
    private String name;
    private Long price;
    //getter/setter
}
public class Author {
    private String name;
    private Integer age;
    //getter/setter
}

此时,如果我定义一个数据添加接口,如下:

@PostMapping("/book")
public void addBook(Book book, Author author) {
    System.out.println(book);
    System.out.println(author);
}

在接收数据传值的时候,两个实体类都有一个name属性,这样的话,同时传name

从前端传递时,无法区分。此时通过@ControllerAdvice 的全局数据预处理可以解决这个问题

步骤如下

1、给接口中的变量取别名

@PostMapping("/book")
public void addBook(@ModelAttribute("b") Book book, @ModelAttribute("a") Author author) {
    System.out.println(book);
    System.out.println(author);
}

2、进行请求数据预处理

在@ControllerAdvice标记的类中添加如下代码:

@InitBinder("b")
public void b(WebDataBinder binder) {
    binder.setFieldDefaultPrefix("b.");
}
@InitBinder("a")
public void a(WebDataBinder binder) {
    binder.setFieldDefaultPrefix("a.");
}

@InitBinder(“b”) 注解表示该方法用来处理和Book和相关的参数,在方法中,给参数添加一个 b 前缀,即请求参数要有b前缀.

3、前端发送请求

请求发送时,通过给不同对象的参数添加不同的前缀,可以实现参数的区分.

如b.name和a.name区分不同的对象中的name

对于@InitBinder,该注解的主要作用是绑定一些自定义的参数。一般情况下我们使用的参数通过@RequestParam,@RequestBody或者@ModelAttribute等注解就可以进行绑定了,

但对于一些特殊类型参数,比如Date,它们的绑定Spring是没有提供直接的支持的,我们只能为其声明一个转换器,将request中字符串类型的参数通过转换器转换为Date类型的参数,从而供给@RequestMapping标注的方法使用。如下是@InitBinder的声明:

给@ControllerAdvice标注的类上

  @InitBinder
  public void globalInitBinder(WebDataBinder binder) {
    binder.addCustomFormatter(new DateFormatter("yyyy-MM-dd"));
  }

添加如上方法

当controller接口中存在参数变量 Date 时,在接受参数之前就会对所有前端穿过来的Date格式数据进行转换。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值