Spring3.0中对异常的处理方法一共提供了两种:一种是使用HandlerExceptionResolver接口;一种是在Controller类内部使用@ExceptionHandler注解。使用第一种方式可以实现全局异常控制,并且Spring已经提供了一个默认的实现类SimpleMappingExceptionResolver;使用第二种方式可以在Controller内部实现更个性化点异常处理方式,灵活性更高。一般来说,项目中只需要采用第一种方式就可以了,每个人都自己定义异常的展现方式,太过个性了,不统一。
从目前的调查结果看,这两种方式不能共存,不知道未来的版本是否能将他们合二为一,这样才能灵活配置。
基于HandlerExceptionResolver接口的异常处理:
使用这种方式只需要实现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/>节点进行声明即可,示例如下:
- <bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
- <!-- 定义默认的异常处理页面,当该异常类型的注册时使用 -->
- <property name="defaultErrorView" value="error"></property>
- <!--定义异常处理页面用来获取异常信息的变量名,默认名value为exception-->
- <property name="exceptionAttribute" value="ex"></property>
- <!-- 定义需要特殊处理的异常,用类名或完全路径名作为key,异常也页名作为值 -->
- <property name="exceptionMappings">
- <props>
- <prop key="IOException">error</prop>
- <prop key="java.sql.SQLException">error</prop>
- </props>
- </property>
- </bean>
基于@ExceptionHandler的异常处理:
该方法需要定义在Controller内部,然后创建一个方法并用@ExceptionHandler注解来修饰用来处理异常,这个方法基本和@RequestMapping修饰的方法差不多,只是可以多一个类型为Exception的参数,@ExceptionHandler中可以添加一个或多个异常的类型,如果为空的话则认为可以触发所有的异常类型错误。
示例代码:
- /**
- * 基于@ExceptionHandler异常处理
- * @author ZYWANG 2011-4-2
- */
- @Controller
- public class ExceptionHandlerController {
- @ExceptionHandler(value={IOException.class,SQLException.class})
- public String exp(Exception ex,HttpServletRequest request) {
- request.setAttribute("ex", ex);
- return "/error.jsp";
- }
- }
页面使用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