系統(tǒng)的 dao拼岳、service枝誊、controller 出現(xiàn)都通過(guò) throws Exception 向上拋出,最后由 springmvc 前端控制器交由異常處理器進(jìn)行異常處理惜纸,如下圖:
常用的springmvc異常處理方式主要有三種:
- 使用系統(tǒng)定義好的異常處理器 SimpleMappingExceptionResolver
- 使用自定義異常處理器
- 使用異常處理注解
異常處理思路:
Controller調(diào)用service叶撒,service調(diào)用dao,異常都是向上拋出的耐版,最終由DispatcherServlet找異常處理器進(jìn)行異常的處理祠够。
一、SimpleMappingExceptionResolver異常處理器
該方式只需要在 SpringMVC 配置文件中注冊(cè)該異常處理器 Bean 即可粪牲。該 Bean 比較特殊哪审,沒(méi)有 id 屬性,無(wú)需顯式調(diào)用或被注入給其它<bean/>虑瀑,當(dāng)異常發(fā)生時(shí)會(huì)自動(dòng)執(zhí)行該類湿滓。
提示:當(dāng)請(qǐng)求參數(shù)的值與接收該參數(shù)的處理器方法形參的類型不匹配時(shí)滴须,會(huì)拋出類型匹
配有誤異常 TypeMismatchException。
1.1自定義異常類
定義三個(gè)異常類:NameException叽奥、AgeException扔水、StudentException。其中 StudentException
是另外兩個(gè)異常的父類朝氓。
StudentException:
public class StudentException extends Exception{
public StudentException(){
super();
}
public StudentException(String message){
super(message);
}
}
NameException:
public class NameException extends StudentException{
public NameException(){
super();
}
public NameException(String message){
super(message);
}
}
AgeException:
public class AgeException extends StudentException{
public AgeException(){
super();
}
public AgeException(String message){
super(message);
}
}
1.2 修改controller
@Controller
@RequestMapping("/test")
public class MyController{
@RequestMapping(value="/register.do")
public ModelAndView register(String name,int age) throws StudentException{
if(!"張三".equals(name)){
throw new NameException("姓名不正確");
}
if(age>50){
throw new AgeException("年齡太大");
}
return new ModelAndView("/WEB-INF/jsp/show.jsp");
}
}
1.3注冊(cè)異常處理器
<!--配置異常處理器-->
<bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
<property name="exceptionMappings">
<props>
<prop key="com.hcx.exception.NameException">/errors/nameErrors.jsp</prop>
<prop key="com.hcx.exception.AgeException">/errors/ageErrors.jsp</prop>
</props>
</property>
<property name="defaultErrorView" value="/errors/defaultErrors.jsp"/>
<property name="exceptionAttribute" value="ex"/>
</bean>
- exceptionMappings:Properties 類型屬性魔市,用于指定具體的不同類型的異常所對(duì)應(yīng)的異常響應(yīng)頁(yè)面。Key 為異常類的全限定性類名赵哲,value 則為響應(yīng)頁(yè)面路徑
- defaultErrorView:指定默認(rèn)的異常響應(yīng)頁(yè)面待德。若發(fā)生的異常不是 exceptionMappings 中指定的異常,則使用默認(rèn)異常響應(yīng)頁(yè)面枫夺。
- exceptionAttribute:捕獲到的異常對(duì)象将宪。一般異常響應(yīng)頁(yè)面中使用。
1.4定義異常響應(yīng)頁(yè)面
在WebRoot下新建一個(gè)目錄errors橡庞,在其中定義三個(gè)異常響應(yīng)頁(yè)面较坛。
二、自定義異常處理器
使用 SpringMVC 定義好的 SimpleMappingExceptionResolver 異常處理器扒最,可以實(shí)現(xiàn)發(fā)生指定異常后的跳轉(zhuǎn)丑勤。但若要實(shí)現(xiàn)在捕獲到指定異常時(shí),執(zhí)行一些操作的目的吧趣,它是完成不了的法竞。此時(shí),就需要自定義異常處理器强挫。自定義異常處理器爪喘,需要實(shí)現(xiàn)HandlerExceptionResolver接口,并且該類需要在SpringMVC配置文件中進(jìn)行注冊(cè)纠拔。
demo1:
2.1定義異常處理器
當(dāng)一個(gè)類實(shí)現(xiàn)了 HandlerExceptionResolver 接口后,只要有異常發(fā)生泛豪,無(wú)論什么異常稠诲,都會(huì)自動(dòng)執(zhí)行接口方法 resolveException()。
public class MyExceptionResolver implements HandlerExceptionResolver{
public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) {
ModelAndView mv = new ModelAndView();
//將異常對(duì)象加入數(shù)據(jù)模型中
mv.addObject("ex",ex);
//設(shè)置默認(rèn)錯(cuò)誤響應(yīng)頁(yè)面
mv.setViewName("/errors/defaultErrors.jsp");
//設(shè)置NameException響應(yīng)頁(yè)面
if(ex instanceof NameException){
mv.setViewName("/errors/nameErrors.jsp");
}
//設(shè)置AgeException響應(yīng)頁(yè)面
if(ex instanceof AgeException){
mv.setViewName("/errors/ageErrors.jsp");
}
return mv;
}
}
2.2注冊(cè)異常處理器
<!--注冊(cè)異常處理器-->
<bean class="com.hcx.exceptionResolvers.MyExceptionResolver"/>
當(dāng)希望在出現(xiàn)異常后诡曙,返回json而不是ModelAndView臀叙,可以通過(guò)response對(duì)象的Writer實(shí)現(xiàn)
public class MyExceptionHandler implements HandlerExceptionResolver {
public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler,
Exception ex) {
Json json = new Json();
json.setSuccess(false);
json.setMessage("出錯(cuò)了");
PrintWriter writer = null;
try {
writer = response.getWriter();
writer.write(new Gson().toJson(json));
} catch (IOException e) {
e.printStackTrace();
}finally {
if(writer!=null)
{
writer.close();
}
}
return null;
}
}
demo2:
UserException:
@Data
@ToString
@AllArgsConstructor
public class UserException extends Exception{
private String message;
}
ExceptionResolver:
package com.hcx.exception;
import org.springframework.lang.Nullable;
import org.springframework.web.servlet.HandlerExceptionResolver;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* 異常處理器
* Created by hongcaixia on 2020/1/3.
*/
public class ExceptionResolver implements HandlerExceptionResolver{
@Nullable
@Override
public ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, @Nullable Object o, Exception e) {
//獲取異常對(duì)象
UserException userException = null;
if(e instanceof UserException){
userException = (UserException) e;
}else {
userException = new UserException("系統(tǒng)正在維護(hù)");
}
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("errorMsg",userException.getMessage());
modelAndView.setViewName("error");
return modelAndView;
}
}
springmvc.xml文件中配置異常處理器:
<!--配置異常處理器-->
<bean id="exceptionResolver" class="com.hcx.exception.ExceptionResolver"></bean>
error.jsp:
<%@ page contentType="text/html;charset=UTF-8" language="java" isELIgnored="false" %>
<html>
<head>
<title>error</title>
</head>
<body>
${errorMsg}
</body>
</html>
頁(yè)面:
<a href="user/testException">異常處理</a>
三、異常處理注解
使用注解@ExceptionHandler 可以將一個(gè)方法指定為異常處理方法价卤。該注解只有一個(gè)可選屬性 value劝萤,為一個(gè) Class<?>數(shù)組,用于指定該注解的方法所要處理的異常類慎璧,即所要匹配的異常床嫌。而被注解的方法跨释,其返回值可以是 ModelAndView、String厌处,或 void鳖谈,方法名隨意,方法參數(shù)可以是 Exception 及其子類對(duì)象阔涉、HttpServletRequest缆娃、HttpServletResponse 等。系統(tǒng)會(huì)自動(dòng)為這些方法參數(shù)賦值瑰排。對(duì)于異常處理注解的用法贯要,也可以直接將異常處理方法注解于 Controller 之中。
@Controller
@RequestMapping("/test")
public class MyController {
@RequestMapping(value="/register.do")
public ModelAndView register(String name,int age) throws StudentException{
if(!"張三".equals(name)){
throw new NameException("姓名不正確");
}
return new ModelAndView("/WEB-INF/jsp/show.jsp");
}
//NameException異常處理
@ExceptionHandler(NameException.class)
public ModelAndView handleNameException(Exception ex){
ModelAndView mv = new ModelAndView();
mv.addObject("ex",ex);
mv.setViewName("/errors/nameErrors.jsp");
return mv;
}
}
不過(guò)椭住,一般不這樣使用崇渗。而是將異常處理方法專門(mén)定義在一個(gè) Controller 中,讓其它Controller 繼承該 Controller 即可函荣。但是显押,這種用法的弊端也很明顯:Java 是“單繼承多實(shí)現(xiàn)”的,這個(gè) Controller 的繼承將這唯一的一個(gè)繼承機(jī)會(huì)使用了傻挂,使得若再有其它類需要繼承乘碑,將無(wú)法直接實(shí)現(xiàn)。
3.1定義異常處理的controller
@Controller
public class MyExceptionResolver {
//NameException異常處理
@RequestHandler(NameException.class)
public ModelAndView handleNameException(Exception ex){
ModelAndView mv = new ModelAndView();
mv.addObject("ex",ex);
mv.setViewName("/errors/nameErrors.jsp");
return mv;
}
//AgeException異常處理
@ExceptionHandler(AgeException.class)
public ModelAndView handleAgeException(Exception ex){
ModelAndView mv = new ModelAndView();
mv.addObject("ex",ex);
mv.setViewName("/errors/ageErrors.jsp");
return mv;
}
//其他異常處理
@ExceptionHandler
public ModelAndView handleOtherException(Exception ex){
ModelAndView mv = new ModelAndView();
mv.addObject("ex",ex);
mv.setViewName("/errors/defaultErrors.jsp");
return mv;
}
}
2.修改controller
讓普通Controller繼承自定義好的異常處理Controller金拒。
@Controller
@RequestMapping("/test")
public class MyController extends MyExceptionResolver{
@RequestMapping(value="/register.do")
public ModelAndView register(String name,int age) throws StudentException{
if(!"張三".equals(name)){
throw new NameException("姓名不正確");
}
if(age>50){
throw new AgeException("年齡太大");
}
return new ModelAndView("/WEB-INF/jsp/show.jsp");
}
}