首頁 > 軟體

詳解Spring MVC的攔截器與例外處理機制

2022-02-28 19:01:44

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");
}

總結

本篇文章就到這裡了,希望能夠給你帶來幫助,也希望您能夠多多關注it145.com的更多內容!     


IT145.com E-mail:sddin#qq.com