在J2EE项目的开发中,不管是对底层的数据库操作过程,还是业务层的处理过程,还是控制层的处理过程,都不可避免会遇到各种可预知的,不可预知的异常需要处理。每个过程都单独处理异常,系统的代码耦合度高,工作量大且不好统一,维护的工作量也很大。那么,能不能将所有类型的异常处理从各处理过程解耦出来,这样既保证了相关处理过程的功能较单一,也实现了异常信息的统一处理和维护?答案是肯定的。下面将介绍使用Spring MVC统一处理异常的解决和实现过程
-
<李>使用Spring MVC提供的SimpleMappingExceptionResolver李>
<李>实现春天的异常处理接口HandlerExceptionResolver自定义自己的异常处理器李>
<李>使用@ExceptionHandler注解实现异常处理李>
<强>(一)SimpleMappingExceptionResolver 强>
使用这种方式具有集成简单,有良好的扩展性,对已有代码没有入侵性等优点,但该方法仅能获取到异常信息,若在出现异常时,对需要获取除异常以外的数据的情况不适用。
@ configuration @EnableWebMvc @ComponentScan (basePackages={" com.balbala.mvc.web "}) 公开课WebMVCConfig延伸WebMvcConfigurerAdapter { @ bean 公共SimpleMappingExceptionResolver SimpleMappingExceptionResolver () { SimpleMappingExceptionResolver b=new SimpleMappingExceptionResolver (); 属性映射=new属性(); mappings.put (“org.springframework.web.servlet。PageNotFound”、“页面- 404”); mappings.put (“org.springframework.dao。DataAccessException”、“数据访问”); mappings.put (“org.springframework.transaction。TransactionException”、“事务失败”); b.setExceptionMappings(映射); 返回b; } }
<强>(二)HandlerExceptionResolver 强>
相比第一种来说,HandlerExceptionResolver能准确显示定义的异常处理页面,达到了统一异常处理的目标
1。定义一个类实现HandlerExceptionResolver接口,这次贴一个自己以前的代码
包com.athena.common.handler; 进口com.athena.common.constants.ResponseCode; 进口com.athena.common.exception.AthenaException; 进口com.athena.common.http.RspMsg; 进口org.slf4j.Logger; 进口org.slf4j.LoggerFactory; 进口org.springframework.web.servlet.HandlerExceptionResolver; 进口org.springframework.web.servlet.ModelAndView; 进口javax.servlet.http.HttpServletRequest; 进口javax.servlet.http.HttpServletResponse; 进口java.io.IOException; 进口java.io.PrintWriter;/* * *由山姆> 进口com.athena.common.handler.GlobalHandlerExceptionResolver; 进口org.springframework.context.annotation.Bean; 进口com.athena.common.handler.GlobalHandlerExceptionResolver; 进口org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;/* * *由山姆> 包com.balabala.poet.base.spring; 进口com.google.common.base.Throwables; 进口com.raiyee.poet.base.exception.MessageException; 进口com.raiyee.poet.base.utils.Ajax; 进口org.slf4j.Logger; 进口org.slf4j.LoggerFactory; 进口org.springframework.core.annotation.AnnotationUtils; 进口org.springframework.http.HttpStatus; 进口org.springframework.web.bind.annotation.ResponseStatus; 进口org.springframework.web.servlet.ModelAndView; 进口javax.servlet.http.HttpServletRequest; 进口javax.servlet.http.HttpServletResponse; 进口java.io.IOException; 进口java.io.PrintWriter; 进口java.util.Date; 公开课BaseGlobalExceptionHandler { 保护静态最终日志记录器=零; 保护静态最终字符串DEFAULT_ERROR_MESSAGE="系统忙,请稍后再试”; 保护ModelAndView handleError (HttpServletRequest点播,HttpServletResponse负责,异常e弦viewName HttpStatus状态){抛出异常 如果(AnnotationUtils.findAnnotation (e.getClass (), ResponseStatus.class) !=null) 把e; 字符串errorMsg=e instanceof MessageException & # 63;e.getMessage (): DEFAULT_ERROR_MESSAGE; 字符串errorStack=Throwables.getStackTraceAsString (e); getlog ()。错误(“请求:{}{}”,req.getRequestURI (), errorStack); 如果(Ajax.isAjax(点播)){ 返回handleAjaxError(负责、errorMsg、状态); } .toString返回handleViewError (req.getRequestURL () (), errorStack, errorMsg, viewName); } 保护ModelAndView handleViewError (errorMessage errorStack字符串url字符串,字符串,字符串viewName) { ModelAndView飞行器=new ModelAndView (); 飞行器。addObject(“例外”,errorStack); 飞行器。addObject (“url”, url); 飞行器。addObject(“消息”,errorMessage); 飞行器。addObject(“时间戳”,新的日期()); mav.setViewName (viewName); 返回飞行器; } 保护ModelAndView handleAjaxError (HttpServletResponse负责、字符串errorMessage HttpStatus状态)抛出IOException { rsp.setCharacterEncoding (“utf - 8”); rsp.setStatus (status.value ()); PrintWriter作家=rsp.getWriter (); writer.write (errorMessage); writer.flush (); 返回null; } 公共记录器getlog () { 返回LoggerFactory.getLogger (BaseGlobalExceptionHandler.class); } }详解春天全局异常处理的三种方式