程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> JSP編程 >> 關於JSP >> 真實項目實踐:快速定位 Spring MVC異常實戰(附源碼下載)

真實項目實踐:快速定位 Spring MVC異常實戰(附源碼下載)

編輯:關於JSP

真實項目實踐:快速定位 Spring MVC異常實戰(附源碼下載)


一、使用@ExceptionHandler進行處理

1.創建異常基類,使用@ExceptionHandler聲明異常處理

BusinessException和SystemException為自定義異常類,代碼如下:

  1. package com.twosnail.exception;  
  2.    
  3. import javax.servlet.http.HttpServletRequest;  
  4. import org.springframework.stereotype.Controller;  
  5. import org.springframework.web.bind.annotation.ExceptionHandler;  
  6.    
  7. @Controller 
  8. public class BasicExController {  
  9.     /**  
  10.      * 基於@ExceptionHandler異常處理基類  
  11.      * @return  
  12.      */ 
  13.     @ExceptionHandler 
  14.     public String exception( HttpServletRequest request , Exception ex ) {  
  15.            
  16.     // 根據不同錯誤轉向不同頁面    
  17.         if( ex instanceof BusinessException ) {  
  18.             return "business-error";    
  19.         }else if( ex instanceof SystemException ) {   
  20.             return "system-error";  
  21.         } else {  
  22.             return "error";    
  23.         }  
  24.     }  

2、使所有需要異常處理的Controller都繼承該類,如下所示:

  1. public class DemoController extends BasicExController {} 

然而,Dao層、Service層、Controller層拋出的異常(BusinessException、SystemException和其它異常)都能准確顯示定義的異常處理頁面,達到了統一異常處理的目標。

總結:使用@ExceptionHandler注解實現異常處理,具有集成簡單、有擴展性好只需要將要異常處理的Controller類繼承於BasicExController即可)、不需要附加Spring配置等優點,但該方法對已有代碼存在入侵性需要修改已有代碼,使繼承於BasicExController),在異常處理時不能獲取除異常以外的數據。

二、SimpleMappingExceptionResolver簡單異常處理器

SimpleMappingExceptionResolver有兩種配置方式,可以按自己需求而定,配置代碼如下:

1、第一種,在Spring的配置文件中,增加以下內容:

在這裡,可以設置跳轉相應頁面。

  1. <bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">  
  2.     <!-- 定義默認的異常處理頁面,當該異常類型的注冊時使用 -->  
  3.     <property name="defaultErrorView" value="error"></property>  
  4.     <!-- 定義異常處理頁面用來獲取異常信息的變量名,默認名為exception -->  
  5.     <property name="exceptionAttribute" value="ex"></property>  
  6.     <!-- 定義需要特殊處理的異常,用類名或完全路徑名作為key,異常也頁名作為值 -->  
  7.     <property name="exceptionMappings">  
  8.         <props>  
  9.             <prop key="com.twosnail.exception.BusinessException">business-error</prop>  
  10.             <prop key="com.twosnail.exception.SystemException">system-error</prop>  
  11.         </props>  
  12.     </property>  
  13.    
  14.     <!-- 相關狀態碼對應的錯誤頁面 -->  
  15.     <property name="statusCodes">  
  16.         <props>  
  17.             <prop key="errors/500">500</prop>  
  18.             <prop key="errors/404">404</prop>  
  19.         </props>  
  20.     </property>  
  21.     <!-- 設置日志輸出級別,不定義則默認不輸出警告等錯誤日志信息 -->  
  22.     <property name="warnLogCategory" value="WARN" />  
  23.     <!-- 默認HTTP狀態碼 -->  
  24.     <property name="defaultStatusCode" value="500" />  
  25. </bean> 

2、第二種,通過自定義java類,繼承SimpleMappingExceptionResolver

然後在Spring的配置。代碼如下:

  1. <bean id="exceptionResolver" class="com.twosnail.exception.MyselfSimpleMappingExceptionResolver">  
  2.     <property name="exceptionMappings">  
  3.         <props>  
  4.             <prop key="com.twosnail.exception.SystemException">error/500</prop>  
  5.             <prop key="com.twosnail.exception.BusinessException">error/errorpage</prop>  
  6.             <prop key="java.lang.exception">error/500</prop>  
  7.         </props>  
  8.     </property>  
  9. </bean> 

java類代碼如下,在這裡可以處理相應邏輯,如下,分別處理了jsp頁面和json數據:

  1. package com.twosnail.exception;  
  2.    
  3. import java.io.IOException;  
  4. import java.io.PrintWriter;  
  5. import javax.servlet.http.HttpServletRequest;  
  6. import javax.servlet.http.HttpServletResponse;  
  7. import org.springframework.web.servlet.ModelAndView;  
  8. import org.springframework.web.servlet.handler.SimpleMappingExceptionResolver;  
  9.    
  10. public class MyselfSimpleMappingExceptionResolver extends SimpleMappingExceptionResolver {  
  11.    
  12.     @Override 
  13.     protected ModelAndView doResolveException(HttpServletRequest request,  
  14.             HttpServletResponse response, Object handler, Exception ex) {  
  15.         // Expose ModelAndView for chosen error view.  
  16.         String viewName = determineViewName(ex, request);  
  17.         if (viewName != null) {// JSP格式返回  
  18.             if (!(request.getHeader("accept").indexOf("application/json") > -1 || (request  
  19.                     .getHeader("X-Requested-With") != null && request  
  20.                     .getHeader("X-Requested-With").indexOf("XMLHttpRequest") > -1))) {  
  21.                 // 如果不是異步請求  
  22.                 // Apply HTTP status code for error views, if specified.  
  23.                 // Only apply it if we're processing a top-level request.  
  24.                 Integer statusCode = determineStatusCode(request, viewName);  
  25.                 if (statusCode != null) {  
  26.                     applyStatusCodeIfPossible(request, response, statusCode);  
  27.                 }  
  28.                 return getModelAndView(viewName, ex, request);  
  29.             } else {// JSON格式返回  
  30.                 try {  
  31.                     PrintWriter writer = response.getWriter();  
  32.                     writer.write(ex.getMessage());  
  33.                     writer.flush();  
  34.                 } catch (IOException e) {  
  35.                     e.printStackTrace();  
  36.                 }  
  37.                 return null;  
  38.    
  39.             }  
  40.         } else {  
  41.             return null;  
  42.         }  
  43.     }  

總結:使用SimpleMappingExceptionResolver進行異常處理,具有集成簡單、有良好的擴展性、對已有代碼沒有入侵性等優點,但方法1僅能獲取到異常信息,若在出現異常時,對需要獲取除異常以外的數據的情況不適用。

三、HandlerExceptionResolver自定義異常

1.在Spring的配置文件中,增加以下內容:

  1. <bean id="exceptionHandler" class="com.twosnail.exception.MyExceptionHandler"/> 

2.添加自定義的MyExceptionHandler類,代碼如下:

在這裡,單獨打印出了異常路徑,便於在日志中查看,在對SystemException異常進行了特殊處理:

  1. package com.twosnail.exception;  
  2.    
  3. import java.util.Map;  
  4.    
  5. import javax.servlet.http.HttpServletRequest;  
  6. import javax.servlet.http.HttpServletResponse;  
  7. import org.springframework.web.servlet.HandlerExceptionResolver;  
  8. import org.springframework.web.servlet.ModelAndView;  
  9. import org.springframework.web.servlet.View;  
  10. import org.springframework.web.servlet.view.RedirectView;  
  11.    
  12. public class MyExceptionHandler implements HandlerExceptionResolver {  
  13.    
  14.     public ModelAndView resolveException( HttpServletRequest request, HttpServletResponse response,   
  15.             Object handler, Exception exception ) {  
  16.            
  17.         System.out.println( "拋出異常】--異常路徑為:" +   
  18.             request.getServletPath() + "\n異常信息】--" +  exception.getMessage() ) ;  
  19.         //如果不是拋出的action業務異常則不處理  
  20.         if( !( exception instanceof SystemException ) ) {  
  21.             return null;  
  22.         }  
  23.            
  24.         final SystemException actionE = (SystemException) exception;         
  25.         ModelAndView model = null;  
  26.         if( actionE.getForwardType() == SystemException.FORWARD ) {  
  27.                 //進入頁面渲染  
  28.                 model = new ModelAndView( actionE.getModelPath(), actionE.getAttributes());  
  29.         } else if( actionE.getForwardType() == SystemException.REDIRECT ) {  
  30.                 model = new ModelAndView( new RedirectView( actionE.getModelPath(), true));  
  31.         } else {  
  32.             //直接返回頁面內容  
  33.             model = new ModelAndView( new View() {  
  34.                 @Override 
  35.                 public void render(Map<String, ?> arg0, HttpServletRequest arg1,  
  36.                         HttpServletResponse arg2) throws Exception {  
  37.                        
  38.                     arg2.setContentType( "text/html" );  
  39.                     arg2.setCharacterEncoding( actionE.getEncode() );  
  40.                     if( actionE.getResponseBody() != null ) {  
  41.                         arg2.getWriter().print( actionE.getResponseBody() );  
  42.                     }  
  43.                 }  
  44.                    
  45.                 @Override 
  46.                 public String getContentType() {  
  47.                     return "text/html; charset=utf-8";  
  48.                 }  
  49.             } );  
  50.         }  
  51.            
  52.         return model;  
  53.     }  

總結:從上面的集成過程可知,使用實現HandlerExceptionResolver接口的異常處理器進行異常處理,具有集成簡單、有良好的擴展性、對已有代碼沒有入侵性等優點。在異常處理時能獲取導致出現異常的對象,有利於提供更詳細的異常處理信息。而SimpleMappingExceptionResolver就是HandlerExceptionResolver的默認實現類。

四、項目截圖

代碼地址:twosnail源碼地址

參考資料:使用Spring MVC統一異常處理實戰

原創作者:兩只蝸牛



  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved