Spring3.0中对异常的处理方法一共提供了两种:一种是使用HandlerExceptionResolver接口;一种是在Controller类内部使用@ExceptionHandler注解。使用第一种方式可以实现全局异常控制,并且Spring已经提供了一个默认的实现类SimpleMappingExceptionResolver;使用第二种方式可以在Controller内部实现更个性化点异常处理方式,灵活性更高。一般来说,项目中只需要采用第一种方式就可以了,每个人都自己定义异常的展现方式,太过个性了,不统一。
从目前的调查结果看,这两种方式不能共存,不知道未来的版本是否能将他们合二为一,这样才能灵活配置。
使用这种方式只需要实现resolveException方法,该方法返回一个ModelAndView对象,在方法内部对异常的类型进行判断,然后常见合适的ModelAndView对象,如果该方法返回了null,则Spring会继续寻找其他的实现了HandlerExceptionResolver接口的Bean。换句话说,Spring会搜索所有注册在其环境中的实现了HandlerExceptionResolver接口的Bean,逐个执行,直到返回了一个ModelAndView对象。
示例代码:
package cn.easier.exception; import java.io.IOException; import java.sql.SQLException; import java.util.HashMap; import java.util.Map; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.springframework.web.servlet.HandlerExceptionResolver; import org.springframework.web.servlet.ModelAndView; import cn.easier.util.StringPrintWriter; public class CustomExceptionHandler implements HandlerExceptionResolver { @Override public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception exception) { System.out.println("CustomExceptionHandler.........................."); Map<String,Object> map = new HashMap<String,Object>(); //我们还需要一个辅助的类StringPrintWriter,因为ex.printStackTrace参数只有个PrintWriter类型的,java自带的StringWriter //不可用,所以我们需要自己实现一个装饰器的StringPrintWriter。 StringPrintWriter strintPrintWriter = new StringPrintWriter(); exception.printStackTrace(strintPrintWriter); map.put("ex", strintPrintWriter.toString());//将错误信息传递给view if(exception instanceof IOException){ return new ModelAndView("IOError",map); //只能是地址,而且是跟spring-servlet配置中的prefix有关系 }else if(exception instanceof SQLException){ return new ModelAndView("SQLError",map); }else if(exception instanceof RuntimeException){ return new ModelAndView("error",map); //只能在request作用域中 } return null; } }
spring 声明:
<bean id="customExceptionHandler" class="cn.easier.exception.CustomExceptionHandler"/>
这个类必须声明到Spring中去,让Spring管理它,你可以使用@Component标签,也可以使用<bean/>节点。为了简单的进行异常处理,Spring提供了SimpleMappingExceptionResolver类,该类实现了HandlerExceptionResolver接口,需要使用时只需要使用<bean/>节点进行声明即可,示例如下:
该方法需要定义在Controller内部,然后创建一个方法并用@ExceptionHandler注解来修饰用来处理异常,这个方法基本和@RequestMapping修饰的方法差不多,只是可以多一个类型为Exception的参数,@ExceptionHandler中可以添加一个或多个异常的类型,如果为空的话则认为可以触发所有的异常类型错误。
示例代码:
页面使用erro.jsp
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ page import="java.lang.Exception" %> <%@ page import="java.io.PrintWriter" %> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Insert title here</title> </head> <body> <%-- //因为SimpleMappingExceptionResolver中的ex 是一个java.lang.Exception类型 <%Exception ex=(Exception)request.getAttribute("ex");%> --%> <H2>Exception:</H2> <%-- <%ex.printStackTrace(new PrintWriter(out));%> --%> <!--<c:out value="${exception}"></c:out>--> <!--map.put("ex", strintPrintWriter.toString());//将错误信息传递给view --> <%=request.getAttribute("ex")%> </body> </html>
引用地址:http://zywang.iteye.com/blog/983801