Java异常分类和统一处理

一、异常分类
       java异常分为"检查"和"非检查"两类,"检查"二字的意思是,代码编译时,编译器会去Check一下有没有进行异常处理(捕获或向上抛),对于归类为需要检查的异常,若没处理,编译就过不去。
       初学时常常想exception为啥要这样分类处理? 后来慢慢明白,异常不过两种:主观和客观,一个大多数情况下可以避免,一个大多数情况下无法避免。
       像NullPointerException这类异常,大多跟程序员素质挂钩(开发好,测试好, 基本不会在系统运行后蹦出来), 基本是可以避免的,java语法当初把它们归类为‘非检查异常',也算给程序员和编译器省了不少事;

       而像IOException这类跟外在环境有关的异常,几乎是不可避免的(指不定哪一天哪一秒网络就挂了),但是当不期而遇时,程序还是要有所作为,所以编译器有必要督促一下程序员,Check一下,看看是否对这些可能不期而至的异常进行了处理。当Exception对象传递到某个节点后,程序就可以执行一些措施了,比如:给用户返回一个提示("系统繁忙,请重试"),给监控平台推送一个异常消息等等。


二、异常的统一返回处理

1、服务器处理

下面列举Tomcat和Nginx为例

Tomcat是Servlet容器,主要处理动态请求,在web.xml下配置,按http返回码或Exception类型来处理:

<error-page>
  <error-code>404</error-code>
  <location>/WEB-INF/views/error/404.jsp</location>
 </error-page>
 
 <error-page>
  <error-code>500</error-code>
  <location>/WEB-INF/views/error/500.jsp</location>
 </error-page> 
 
 <error-page>
  <exception-type>java.lang.Throwable</exception-type>
  <location>/WEB-INF/views/error/throwable.jsp</location>
 </error-page>

Nginx是反向代理服务器,通过Http返回码也可以很方便地指定异常时的返回页面:

server {  
    listen 80 ;  
    server_name xx.com ;  
    root /var/www ;  
    index index.html ;  
    error_page 404  /404.html ;  
    location = /404.html {  
       root /usr/share/nginx/html;  
    }  
}  
2、框架处理

下面列举Spring MVC的处理方式

(1)使用Spring MVC自带的简单异常处理器SimpleMappingExceptionResolver; 
(2)实现接口HandlerExceptionResolver 自定义异常处理器; (建议使用,可支持ajax等扩展)
(3)使用@ExceptionHandler注解实现异常处理; 

第(1)种,在spring-mvc.xml下配置

<!-- 将Controller抛出的异常转到特定视图 -->
    <bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
        <property name="exceptionMappings">
            <props>
             <!-- 不同异常分开跳转-->    
             <!-- 可以自定义不同的异常-->              
                <prop key="com.test.MyException1">/error/e1</prop>
                <prop key="com.test.MyException2">/error/e2</prop>
                <!-- 如果不想自定义异常,只配置下面的即可--> 
                <prop key="java.lang.Throwable">/error/500</prop>
            </props>
        </property>
    </bean>

缺点:无法处理不需要返回html的请求;


 第(2)种,自定义HandlerExceptionResolver接口的实现类

/**
 * 自定义异常处理器:支持ajax
 * 
 */
public class MyExceptionHandler implements HandlerExceptionResolver {
 public ModelAndView resolveException(HttpServletRequest request,
   HttpServletResponse response, Object handler, Exception ex) {
  /* 区分ajax */
  boolean isAjax = request.getHeader("X-Requested-With") != null
    && "XMLHttpRequest".equals(request
      .getHeader("X-Requested-With").toString());
  if (!isAjax) {
   if (ex instanceof com.test.MyException1) {
    return new ModelAndView("/error/e1");
   } else if (ex instanceof com.test.MyException1) {
    return new ModelAndView("/error/e2");
   } else {
    return new ModelAndView("/error/500");
   }
  }
  String jsonRes = "{\"message\":\"" + "系统异常" + "\"}";// 自定义结构和前台对接
  PrintWriter out = null;
  try {
   response.setCharacterEncoding("utf-8");
   response.setContentType("application/json;charset=UTF-8");
   out = response.getWriter();
   out.print(jsonRes);
   out.flush();
  } catch (IOException e) {
   e.printStackTrace();
  } finally {
   out.close();
  }
  return null;
 }
}

并在spring-mvc.xml下注册处理器

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

优点:可以处理ajax请求,也方便编码实现功能扩展,比如异常的监控等。

第(3)种,@ExceptionHandler注解

@Controller
public class TestExceptionHandlerController {
 
 @ExceptionHandler({ MyException1.class })
 public String exception(MyException1 e) {
  return "/error/e1";
 }
 @RequestMapping("/marry")
 public void test() {
  throw new MyException1("有问题");
 }
}

缺点:@ExceptionHandler的方法,必须和可能抛异常的方法在一同个Controller下。(不建议使用)

PS:实际情况,会将返回码的映射交给服务器,将动态请求的一些自定义异常或者功能扩展交给框架。



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值