概述:
只要程序都需要处理异常,Spring会有哪些处理异常的方式
- 特定的Spring异常将会自动映射为指定的HTTP状态码
- 异常上可以添加@ResponseStatus注解,从而改变HTTP状态码
- 在控制器中添加一个带有@ExceptionHandler注解可以处理这个控制器所有抛出此异常的
对于自动映射这里不列举了
2、异常类添加@RepsonseStatus注解
首先创建一个自定义异常类:
package com.jack.controller;
import org.springframework.http.HttpStatus;
import org.springframework.web.bind.annotation.ResponseStatus;
@ResponseStatus(value=HttpStatus.NOT_FOUND,
reason="Spittle Not Found")
public class SpittleNotFoundException extends RuntimeException {
/**
*
*/
private static final long serialVersionUID = -4110600977121137282L;
}
总结:可以看到在异常类上添加@ResponseStatus注解,value为状态码,reason描述
然后将该类应用到controller类中
@RequestMapping(value="/{spittleId}", method=RequestMethod.GET)
public String spittle(@PathVariable("spittleId") long spittleId,
Model model) throws SpittleNotFoundException{
Spittle spittle = spittleRepository.findOne(spittleId);
spittle=null;
if(spittle == null) {
throw new SpittleNotFoundException();
}
model.addAttribute(spittle);
return "spittle";
}
故意让spittle 为null
效果如下:如果不加这个注解,会抛出500异常,同时会打印堆栈的信息
3、一些优秀的网址都会处理404页面,而且还很漂亮,如何处理这个控制器所有抛出SpittleNotFoundException
接下来用到的注解就是@ExceptionHandler
只要在控制器类中添加如下方法:
@ExceptionHandler(SpittleNotFoundException.class)
public String handleDuplicateSpittle(){
return "notfound";
}
总结:对于所有SpittleNotFoundException 都会被这个方法处理
这里采用thymeleaf框架,需要建立一个notfound.html文件
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:th="http://www.thymeleaf.org"> <!-- 声明Thymeleaf命名空间 -->
<head>
<title>Spittr</title>
<link rel="stylesheet" type="text/css"
th:href="@{/resource/style.css}"></link>
</head>
<body>
<h1>对不起 Spittle 迷路了</h1>
</body>
</html>
4、问题又来了,如果有很多controller类,岂不重复写很多遍,这又违背dry原则,如何实现一次编辑,多地使用---为控制器添加通知
package com.jack.config;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import com.jack.controller.SpittleNotFoundException;
@ControllerAdvice
public class NotFoundExceptionHandler {
@ExceptionHandler(SpittleNotFoundException.class)
public String handleDuplicateSpittle(){
return "notfound";
}
}
总结:只是在增加了ControllerAdvice 注解,最后的效果就是所有Controller抛出此异常都会在这里处理。