詳解Spring MVC的攔截器與異常處理機制

1. SpringMVC攔截器

1.1 攔截器(interceptor)的作用

Spring MVC 的攔截器類似於 Servlet 開發中的過濾器 Filter,用於對處理器進行預處理後處理

將攔截器按一定的順序聯結成一條鏈,這條鏈稱為攔截器鏈(Interceptor Chain)。在訪問被攔截的方

法或字段時,攔截器鏈中的攔截器就會按其之前定義的順序被調用。攔截器也是AOP思想的具體實現。

1.2 攔截器和過濾器的區別

區別 過濾器(Filter) 攔截器(Interceptor)
使用范圍 是servlet規范中的一部分,任何Java Web工程都可以使用 是Spring MVC框架自己的,隻有使用瞭Spring MVC框架的工程才可以使用
攔截范圍 在url-pattern 中配置瞭/*後,可以對所有要訪問的資源攔截 在<mvc:mapping path=" “/>中配置瞭/**之後,也可以對所有資源進行攔截,但是可以通過<mvc:exclude-mapping path=” "/>標簽 排除不需要攔截的資源

1.3 攔截器的快速入門

自定義攔截器很簡單,隻有如下三步:

1.創建攔截器類 實現HandlerInterceptor接口

2.配置攔截器

3.測試攔截器的攔截效果

創建攔截器類 實現HandlerInterceptor接口

public class MyHandlerInterceptor implements HandlerInterceptor{
    public boolean preHandle(HttpServletRequest request,HttpServletResponse response,Object handler){
        System.out.println("preHandle running...");
        return true;
    }
    
    public boolean postHandler(HttpServletRequest request,HttpServletResponse response,Object handler){
        System.out.println("postHandler running....");
    }
    public void afterCompletion(HttpServletRequest request,HttpServletResponse response,Object handler){
        System.out.println("aferCompletion running...");
    }
}

配置攔截器

<!-- 配置攔截器-->
<mvc:interceptors>
	<mvc:interceptor>
    	<mvc:mapping path="/**"/>
        <bean class = "com.cs.interceptor.MyHandlerInterceptor"/>
    </mvc:interceptor>
</mvc:interceptors>

測試攔截器的攔截效果\

@ResponseMapping("/test1")
@ResponseBody
public ModelAndView quickMethod() throws Exception{
    System.out.println("目標方法執行");
    ModelAndView modelAndView = new ModelAndView();
    modelAndView.addObject("name","bestcollc");
    modelAndView.setViewName("index");
    return modelAndView;
}

測試結果:
    preHandle running....
    目標方法執行
    postHandle running...
    afterCompletion running...

1.4 多攔截器操作

同上,在編寫一個MyHandlerInterceptor2操作,測試執行順序

1.5 攔截器方法說明

方法名 說明
preHandle() 方法將在請求處理之前進行調用,該方法的返回值是佈爾值Boolean類型的,當它返回為false 時,表示請求結束,後續的Interceptor 和Controller 都不會再執行;當返回值為true 時就會繼續調用下一個Interceptor 的preHandle 方 法
postHandle() 該方法是在當前請求進行處理之後被調用,前提是preHandle 方法的返回值為true 時才能被調用,且它會在DispatcherServlet 進行視圖返回渲染之前被調用,所以我們可以在這個方法中對Controller 處理之後的ModelAndView 對象進行操作
afterCompletion() 該方法將在整個請求結束之後,也就是在DispatcherServlet 渲染瞭對應的視圖之後執行,前提是preHandle 方法的返回值為true 時才能被調用

2. SpringMVC異常處理

2.1 異常處理的思路

系統中異常包括兩類:預期異常運行時異常RuntimeException,前者通過捕獲異常從而獲取異常信息,後者主要通過規范代碼開發、測試等手段減少運行時異常的發生。

系統的DaoServiceController出現都通過throws Exception向上拋出,最後由SpringMVC前端控制器交

由異常處理器進行異常處理,如下圖:

在這裡插入圖片描述

2.2 異常處理的兩種方式

  • 使用Spring MVC提供的簡單異常處理器 SimpleMappingExceptionResolver
  • 實現Spring 的異常處理接口HandlerExcepionResolver自定義自己 的異常處理器

2.3 簡單的異常處理器SimpleMappingExceptinResolver

SpringMVC已經定義好瞭該類型轉換器,在使用時可以根據項目情況進行相應異常與視圖的映射配置

<!--配置簡單映射異常處理器 -->
<bean
      class = "org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
		<property	name="defaultErrorView" value="error"/>   默認錯誤視圖
    	<property name="exceptionMappings"
    			<map>
    					<entry key="com.cs.exception.Myexception" value="error"/>
                    	<entry key="java.lang.ClassCastException" value="error"/>
    			</map>
		</property>
</bean>

2.4 自定義異常處理步驟

  • 創建異常處理器類實現HandlerExceptionResolver
  • 配置異常處理器
  • 編寫異常頁面
  • 測試異常跳轉

1.創建異常處理器類實現HandlerExceptionResolver

public class MyExceptionResolver implements HandlerExceptionResolver {
	@Override
    public ModelAndView resolveException(HttpServletRequest request, 
    HttpServletResponse response, Object handler, Exception ex) {
        //處理異常的代碼實現
        //創建ModelAndView對象
        ModelAndView modelAndView = new ModelAndView();
        modelAndView.setViewName("exceptionPage");
        return modelAndView;
	} 
}

2.配置異常處理器

<bean id="exceptionResolver"
      	class="com.cs.exception.MyExceptionResolver"/>

3.編寫異常頁面

<%@pagecontentType="text/html;charset=UTF-8"language="java"%>
<html>
<head>
    	<title>Title</title>
</head>
<body>
    	這是一個最終異常的顯示頁面 
</body>
</html>

4.測試異常跳轉

@RequestMapping("/quick22")
@ResponseBody 
publicvoidquickMethod22()throwsIOException,ParseException{
    SimpleDateFormatsimpleDateFormat=newSimpleDateFormat("yyyy-MM-dd");
    simpleDateFormat.parse("abcde");
}

總結

本篇文章就到這裡瞭,希望能夠給你帶來幫助,也希望您能夠多多關註WalkonNet的更多內容!     

推薦閱讀: