chapter07_SpringMVC的高级技术_3_处理异常

  • 无论在服务器端的程序是否出现异常,有一个request过来时,总会有一个response响应回去。当内部不做任何处理时,如果出现异常,就会映射成 500 Internal Server Error

  • 有时,由于路径错误等问题造成了空指针,产生NullPointerException,如果不做处理,也会输出 500 错误;而事实上这是由于没有找到资源文件造成的。我们希望它输出404 Not Found。这需要__更加精细__的异常处理

  • 抛出更加精细的异常

    示例 SpittleNotFoundException.java

      @ResponseStatus(value = HttpStatus.NOT_FOUND, reason = "Spittle Not Found")
      public class SpittleNotFoundException extends RuntimeException {
      }
    

    添加了 @ResponseStatus 注解后,如果某个控制器方法抛出了 SpittleNotFoundException 这一运行期异常时,不会响应为500,而是会响应为404(HttpStatus.NOT_FOUND)

  • 处理异常

    (1) 有时,需要根据异常类型,返回我们编写好的异常页面。而且虽然几种异常对应的都是404 error,但是它们可能需要返回不同的页面。这就需要在控制器方法中catch这些异常,并对它们进行处理

    示例 SpittleController.java

      @Controller
      @RequestMapping("/spittles")
      public class SpittleController {
    
          ...
    
          @RequestMapping(method = RequestMethod.POST)
          public String saveSpittle(SpittleForm form, Model model) {
    
              try {
          
                  spittleRepository.save(new Spittle(null, form.getMessage(), new Date(), form.getLongitude(), form.getLatitude()));
    
                  return "redirect:/spittles";
    
              } catch (DuplicateSpittleException e) {
    
                  return "error/duplicate";
          
              } catch (RuntimeException e) {
    
                  return "error/sql";
          }
    
          ...
      }
    

    这样做可以满足需求,但是当需要处理的异常情况很多时,会扰乱主体控制器方法的编写

    (2) 在控制器类中添加异常处理方法

    示例 SpittleController.java

      @Controller
      @RequestMapping("/spittles")
      public class SpittleController {
    
          ...
    
          @RequestMapping(method = RequestMethod.POST)
          public String saveSpittle(SpittleForm form, Model model) {
    
              spittleRepository.save(new Spittle(null, form.getMessage(), new Date(), form.getLongitude(), form.getLatitude()));
    
              return "redirect:/spittles";
          }
    
          @ExceptionHandler(DuplicateSpittleException.class)
          public String handleNotFound() {
    
              return "error/duplicate";
          }
    
          ...
      }
    

    在控制器类中增加带有 @ExceptionHandler注解的方法,它可以处理控制器内__所有方法__中抛出的DuplicateSpittleException异常。

    但是,有的时候不同的控制器可能会抛出相同的异常类型,它们应该对应着相同的异常页面;采用这种方法,会导致不同的控制器中出现几乎相同的异常处理方法

    (3) 为控制器添加通知

    运用AOP的思想,可以写一个专门的类,用来处理所有的相同类型的异常。

    示例 AppWideExceptionHandler.java

      @ControllerAdvice
      public class AppWideExceptionHandler {
    
          @ExceptionHandler(DuplicateSpittleException.class)
          public String handleDuplicateSpittle() {
    
              return "error/duplicate";
          }
    
          @ExceptionHandler(RuntimeException.class)
          public String handleOtherRuntimeException() {
    
              return "error/otherException";
          }
      }
    

    带有@ControllerAdvice的类通知所有的带有@RequestMapping注解的控制器上,这个类要包含1个或__多个__带有@ExceptionHandler注解的方法;

    @ControllerAdvice已经使用了@Component注解,所以会被@ComponentScan到

    添加了@ExceptionHandler方法和控制器中添加了@RequestMapping的方法的内容相似,也是要返回一个String类型的逻辑视图名称

猜你喜欢

转载自blog.csdn.net/captxb/article/details/87880153