zoukankan      html  css  js  c++  java
  • spring mvc 全局处理异常

    spring框架支持很多种全局处理异常的方式

    一、Spring MVC处理异常有4种方式: 
    (1)使用Spring-MVC提供的SimpleMappingExceptionResolver; 
    (2)实现Spring的异常处理接口HandlerExceptionResolver 自定义自己的异常处理器; 
    (3)使用@ExceptionHandler注解实现异常处理; 

    (4)使用@ControllerAdvice方式

    二、分别介绍这三种异常处理的实现方式:

    (1)使用SimpleMappingExceptionResolver实现异常处理

              只需要在Spring的配置文件applicationContext.xml中增加以下内容:

    <bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">  
        <!-- 定义默认的异常处理页面,当该异常类型的注册时使用 -->  
        <property name="defaultErrorView" value="error"></property>  
        <!-- 定义异常处理页面用来获取异常信息的变量名,默认名为exception -->  
        <property name="exceptionAttribute" value="ex"></property>  
        <!-- 定义需要特殊处理的异常,用类名或完全路径名作为key,异常以页名作为值 -->  
        <property name="exceptionMappings">  
            <props>  
                <!-- 创建自己所要自定义的异常类 -->
                <prop key="com.core.exception.BusinessException">business_error</prop>  
                <prop key="com.core.exception.ParameterException">parameter_error</prop>  
                <!-- 还可以继续扩展对不同异常类型的处理 -->  
            </props>  
        </property>  
    </bean>
    

      

    (2) 实现HandlerExceptionResolver 接口自定义异常处理器

      首先增加HandlerExceptionResolver 接口的实现类MyExceptionHandler,代码如下: 

    public class MyExceptionHandler implements HandlerExceptionResolver {  
      
        public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, 
            Object handler, Exception ex) {  
            Map<String, Object> model = new HashMap<String, Object>();  
            model.put("ex", ex);  
            // 根据不同错误转向不同页面  
            if(ex instanceof BusinessException) {  
                return new ModelAndView("business_error", model);  
            }else if(ex instanceof ParameterException) {  
                return new ModelAndView("parameter_error", model);  
            } else {  
                return new ModelAndView("error", model);  
            }  
        }  
    }
    

      

        然后在Spring的配置文件applicationContext.xml中增加以下内容:

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

      

       (3)使用@ExceptionHandler注解实现异常处理 

         首先要增加BaseController类,并在类中使用@ExceptionHandler注解声明异常处理,代码如下:

    public class BaseController {  
        /** 基于@ExceptionHandler异常处理 */  
        @ExceptionHandler  
        public String exp(HttpServletRequest request, Exception ex) {  
            request.setAttribute("ex", ex);  
            // 根据不同错误转向不同页面  
            if(ex instanceof BusinessException) {  
                return "business_error";  
            }else if(ex instanceof ParameterException) {  
                return "parameter_error";  
            } else {  
                return "error";  
            }  
        }  
    }
    

      

        然后需要修改现有代码,使所有需要异常处理的Controller都继承该类,如下所示:

    public class TestController extends BaseController


    (4) 使用@ControllerAdvice

    @ControllerAdvice(annotations = Controller.class)
    public class GlobalExceptionHandler {
    
        @ExceptionHandler(SQLException.class)
        @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
        @ResponseBody
        public RespBody handleSQLException(HttpServletRequest request, Exception ex) {
            String message = ex.getMessage();
            return  RespBody.error(HttpStatus.INTERNAL_SERVER_ERROR.value(), message);
        }
      
     @ExceptionHandler(Exception.class)
     @ResponseBody
     public WapRespBody handleActivityException(HttpServletRequest request, Exception ex) {
         writeLog(ex,request);
         return  WapRespBody.error();
     }
    
    }
    

      

    自定义返回类RespBody、异常类、异常枚举等

    首先自定义返回类:
    //作用于类或属性上,被用来在序列化/反序列化时为该对象或字段添加一个对象识别码,通常是用来解决循环嵌套的问题
    @JsonIdentityInfo(generator = ObjectIdGenerators.IntSequenceGenerator.class,property = "@code")
    public class WapRespBody {
    
    
        private String state;
    
        private String message;
    
        //正常数据
        private Object data;
    
    
        private WapRespBody() {
    
        }
    
        private WapRespBody(String code, String message) {
            this.state = code;
            this.message = message;
        }
    
        private WapRespBody(String statusCode, Object data) {
            this.state = statusCode;
            this.data = data;
        }
    
        private WapRespBody(String statusCode, String message, Object data) {
            this.state = statusCode;
            this.message = message;
            this.data = data;
        }
    
        public static WapRespBody ok() {
            return new WapRespBody("200", "");
        }
    
        public static WapRespBody ok(Object data) {
            return new WapRespBody("200", data);
        }
    
        public static WapRespBody ok(String code, String message) {
            return new WapRespBody(code, message);
        }
    
    
        public static WapRespBody error() {
            return new WapRespBody("-1", "");
        }
    
        public static WapRespBody error(Object data) {
            return new WapRespBody("-1", data);
        }
    
        public static WapRespBody error(String code, String msg) {
            return new WapRespBody(code, msg);
        }
    
        public String getState() {
            return state;
        }
    
        public void setState(String state) {
            this.state = state;
        }
    
        public String getMessage() {
            return message;
        }
    
        public void setMessage(String message) {
            this.message = message;
        }
    
        public Object getData() {
            return data;
        }
    
        public void setData(Object data) {
            this.data = data;
        }
    }

    自定义异常信息

    package com.yunqiandai.common.exception;
    
    import com.yunqiandai.common.constant.ActivityEnum;
    
    /**
     * Created with IntelliJ IDEA.
     *
     * @author: zhaojc
     * @date: 16/12/13
     * @Time: 上午11:41
     * @description:
     */
    public class ActivityException extends Exception {
    
        private String errorCode;
    
        private String errorMessage;
    
    
        public ActivityException() {
    
        }
    
        public ActivityException(String code, String message) {
            this.errorCode = code;
            this.errorMessage = message;
        }
    
        public ActivityException(ActivityEnum activityEnum) {
            this.errorCode = activityEnum.getStatus();
            this.errorMessage = activityEnum.getMessage();
        }
    
        public String getErrorCode() {
            return errorCode;
        }
    
        public void setErrorCode(String errorCode) {
            this.errorCode = errorCode;
        }
    
        public String getErrorMessage() {
            return errorMessage;
        }
    
        public void setErrorMessage(String errorMessage) {
            this.errorMessage = errorMessage;
        }
    }
    

      

    自定义枚举

    package com.yunqiandai.common.constant;
    
    /**
     * Created with IntelliJ IDEA.
     * @author: zhaojc
     * @date: 16/12/1
     * @Time: 上午10:19
     * @description: 活动常量
     */
    public enum ActivityEnum {
    
        //用户未登录
        USER_NO_LOGIN("201","用户未登录"),
    
        //活动已开始
        ACTIVITY_IS_START("202","活动已开始"),
    
        //活动未开始
        ACTIVITY_NO_START("203","活动未开始"),
    
        //参数错误
        ACTIVITY_PARAM_ERROR("204","参数错误"),
    
        //页面返回正常
        RESULT_CODE_OK("00",""),
        //页面返回失败
        RESULT_CODE_ERROR("01",""),
        //系统繁忙、发送失败
        RESULT_STATE_ERROR("0","系统繁忙、发送失败!"),
        //请先绑定手机号、请重新登录
        RESULT_STATE_ERR("-1","请先绑定手机号、请重新登录!"),
        //state正常
        RESULT_STATE_OK("1","");
    
        private String status;
    
        private String message;
    
        ActivityEnum(String status,String message) {
            this.status = status;
            this.message=message;
        }
    
        public String getStatus() {
            return status;
        }
    
        public void setStatus(String status) {
            this.status = status;
        }
    
    
        public String getMessage() {
            return message;
        }
    
        public void setMessage(String message) {
            this.message = message;
        }
    }
    

      

    三、未捕获异常的处理 :
            对于Unchecked Exception而言,由于代码不强制捕获,往往被忽略,如果运行期产生了UncheckedException,而代码中又没有进行相应的捕获和处理,则我们可能不得不面对尴尬的404、500……等服务器内部错误提示页面。 
            我们需要一个全面而有效的异常处理机制。目前大多数服务器也都支持在Web.xml中通过<error-page>(Websphere/Weblogic)或者<error-code>(Tomcat)节点配置特定异常情况的显示页面。

    实现方式如下:

    修改web.xml文件,增加以下内容: 

    <!-- 出错页面定义 -->  
    <error-page>  
        <exception-type>java.lang.Throwable</exception-type>  
        <location>/500.jsp</location>  
    </error-page>  
    <error-page>  
        <error-code>500</error-code>  
        <location>/500.jsp</location>  
    </error-page>  
    <error-page>  
        <error-code>404</error-code>  
        <location>/404.jsp</location>  
    </error-page>  
    <!-- 可继续增加服务器错误号的处理及对应显示页面 -->
    

      

    四、比较异常处理方式的优缺点:

            Spring MVC集成异常处理3种方式都可以达到统一异常处理的目标。从4种方式的优缺点比较,若只需要简单的集成异常处理,推荐使用SimpleMappingExceptionResolver即可;若需要集成的异常处理能够更具个性化,提供给用户更详细的异常信息,推荐自定义实现HandlerExceptionResolver接口的方式;若不喜欢Spring配置文件或要实现“零配置”,且能接受对原有代码的适当入侵,则建议使用@ExceptionHandler注解方式。 

    如果异常处理配置在项目后期的话,需要看下项目是否用到过以上方式统一管理异常,否则会出现配置后失效的问题。

  • 相关阅读:
    POJ 1659 Frogs' Neighborhood (贪心)
    HDU 2544 最短路 (Floyd)
    CodeForces 632C Grandma Laura and Apples (模拟)
    CodeForces 731F Video Cards (数论+暴力)
    CodeForces 731C Socks (DFS或并查集)
    CodeForces 731B Coupons and Discounts (水题模拟)
    CodeForces 731A Night at the Museum (水题)
    UVaLive 6834 Shopping (贪心)
    zzuli 1484 继续双线
    zzuli 1875多线DP
  • 原文地址:https://www.cnblogs.com/moon521/p/6179333.html
Copyright © 2011-2022 走看看