七、SpringMVC中的异常处理
(1)异常处理的思路
系统中异常包括两类:预期异常和运行时异常 RuntimeException,前者通过捕获异常从而获取异常信息,后者主要通过规范代码开发、测试通过手段减少运行时异常的发生。
系统的 dao、service、controller 出现都通过 throws Exception 向上抛出,最后由 springmvc 前端控制器交由异常处理器进行异常处理,如下图:
(2)实现步骤
1、编写异常类和错误页面
//自定义异常类
public class SysException extends Exception{
//存储提示信息的
private String message;
//构造方法
public SysException(String message) {
this.message = message;
}
public String getMessage() {
return message;
}
public void setMessage(String message) {
this.message = message;
}
}
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
${errorMsg}
</body>
</html>
2、自定义异常处理器
public class SysExceptionResolver implements HandlerExceptionResolver {
@Override
public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) {
//获取到异常对象
SysException e=null;
if (ex instanceof SysException){
e= (SysException) ex;
}else{
e = new SysException("系统正在维护......");
}
//创建ModelAndView对象
ModelAndView mv = new ModelAndView();
mv.addObject("error",e.getMessage());
mv.setViewName("error");
return mv;
}
}
3、配置异常处理器
<!--配置异常处理器-->
<bean id="sysExceptionResolver" class="org.westos.demo.exception.SysExceptionResolver"></bean>
4、运行结果
八、SpringMVC中的拦截器
(1)拦截器的作用
Spring MVC 的处理器拦截器类似于 Servlet 开发中的过滤器 Filter,用于对处理器进行预处理和后处理。
用户可以自己定义一些拦截器来实现特定的功能。
谈到拦截器,还要向大家提一个词——拦截器链(Interceptor Chain)。拦截器链就是将拦截器按一定的顺序联结成一条链。在访问被拦截的方法或字段时,拦截器链中的拦截器就会按其之前定义的顺序被调用。
说到这里,可能大家脑海中有了一个疑问,这不是我们之前学的过滤器吗?是的它和过滤器是有几分相似,但是也有区别,接下来我们就来说说他们的区别:
过滤器是 servlet 规范中的一部分,任何 java web 工程都可以使用。
拦截器是 SpringMVC 框架自己的,只有使用了 SpringMVC 框架的工程才能用。
过滤器在 url-pattern 中配置了/*之后,可以对所有要访问的资源拦截。
拦截器它是只会拦截访问的控制器方法,如果访问的是 jsp,html,css,image 或者 js 是不会进行拦截的。
它也是 AOP 思想的具体应用。
我们要想自定义拦截器, 要求必须实现:HandlerInterceptor 接口。
(2)自定义拦截器的步骤
1、编写一个HandlerInterceptor接口
//自定义拦截器
public class MyInterceptor1 implements HandlerInterceptor {
/*
* 预处理,controller方法执行前
* return true放行,执行下一个拦截器,如果没有,执行controller中的方法
* return false不放行
* */
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("MyInterceptor1执行了......前111(第一个拦截器)");
//request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
//true为放行,false为不放行
return true;
}
/*
* 后处理方法
* controller方法执行后
* success.jsp执行之前
* */
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("MyInterceptor1执行了......后111(第一个拦截器)");
request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
}
/*
* success.jsp页面执行后,该方法会执行
* 这个页面不能再跳转其他页面了
* */
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("MyInterceptor1执行了......最后111(第一个拦截器)");
}
}
2、配置拦截器
<!--配置拦截器-->
<mvc:interceptors>
<!--配置第一个拦截器-->
<mvc:interceptor>
<!--要拦截的具体的方法-->
<mvc:mapping path="/user/*"/>
<!--不要拦截的方法-->
<!--<mvc:exclude-mapping path=""/>-->
<!--配置拦截器对象-->
<bean class="org.westos.demo.interceptor.MyInterceptor1"/>
</mvc:interceptor>
(3)拦截器的细节
1、拦截器的放行
放行的含义是:如果有下一个拦截器就执行下一个,如果该拦截器处于拦截器链的最后一个,则执行控制器中的方法。
2、拦截器的作用路径
作用路径可以通过在配置文件中配置。
<!--配置拦截器-->
<mvc:interceptors>
<!--配置第一个拦截器-->
<mvc:interceptor>
<!--要拦截的具体的方法-->
<mvc:mapping path="/user/*"/>
<!--不要拦截的方法-->
<!--<mvc:exclude-mapping path=""/>-->
<!--配置拦截器对象-->
<bean class="org.westos.demo.interceptor.MyInterceptor1"/>
</mvc:interceptor>
<!--配置第二个拦截器-->
<mvc:interceptor>
<!--要拦截的具体的方法-->
<mvc:mapping path="/**/"/>
<!--不要拦截的方法-->
<!--<mvc:exclude-mapping path=""/>-->
<!--配置拦截器对象-->
<bean class="org.westos.demo.interceptor.MyInterceptor2"/>
</mvc:interceptor>
</mvc:interceptors>
3、多个拦截器执行的顺序
多个拦截器是按照配置的顺序决定的。
(4)正常流程测试
配制文件
<!--配置拦截器-->
<mvc:interceptors>
<!--配置第一个拦截器-->
<mvc:interceptor>
<!--要拦截的具体的方法-->
<mvc:mapping path="/user/*"/>
<!--不要拦截的方法-->
<!--<mvc:exclude-mapping path=""/>-->
<!--配置拦截器对象-->
<bean class="org.westos.demo.interceptor.MyInterceptor1"/>
</mvc:interceptor>
<!--配置第二个拦截器-->
<mvc:interceptor>
<!--要拦截的具体的方法-->
<mvc:mapping path="/**/"/>
<!--不要拦截的方法-->
<!--<mvc:exclude-mapping path=""/>-->
<!--配置拦截器对象-->
<bean class="org.westos.demo.interceptor.MyInterceptor2"/>
</mvc:interceptor>
</mvc:interceptors>
拦截器1的代码
//自定义拦截器
public class MyInterceptor1 implements HandlerInterceptor {
/*
* 预处理,controller方法执行前
* return true放行,执行下一个拦截器,如果没有,执行controller中的方法
* return false不放行
* */
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("MyInterceptor1执行了......前111(第一个拦截器)");
//request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
//true为放行,false为不放行
return true;
}
/*
* 后处理方法
* controller方法执行后
* success.jsp执行之前
* */
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("MyInterceptor1执行了......后111(第一个拦截器)");
request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
}
/*
* success.jsp页面执行后,该方法会执行
* 这个页面不能再跳转其他页面了
* */
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("MyInterceptor1执行了......最后111(第一个拦截器)");
}
}
拦截器2的代码
//自定义拦截器
public class MyInterceptor2 implements HandlerInterceptor {
/*
* 预处理,controller方法执行前
* return true放行,执行下一个拦截器,如果没有,执行controller中的方法
* return false不放行
* */
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("MyInterceptor1执行了......前222(第二个拦截器)");
//request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
//true为放行,false为不放行
return true;
}
/*
* 后处理方法
* controller方法执行后
* success.jsp执行之前
* */
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("MyInterceptor1执行了......后222(第二个拦截器)");
//request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
}
/*
* success.jsp页面执行后,该方法会执行
* 这个页面不能再跳转其他页面了
* */
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("MyInterceptor1执行了......最后222(第二个拦截器)");
}
}
(5)中断流程测试
配制文件
<!--配置拦截器-->
<mvc:interceptors>
<!--配置第一个拦截器-->
<mvc:interceptor>
<!--要拦截的具体的方法-->
<mvc:mapping path="/user/*"/>
<!--不要拦截的方法-->
<!--<mvc:exclude-mapping path=""/>-->
<!--配置拦截器对象-->
<bean class="org.westos.demo.interceptor.MyInterceptor1"/>
</mvc:interceptor>
<!--配置第二个拦截器-->
<mvc:interceptor>
<!--要拦截的具体的方法-->
<mvc:mapping path="/**/"/>
<!--不要拦截的方法-->
<!--<mvc:exclude-mapping path=""/>-->
<!--配置拦截器对象-->
<bean class="org.westos.demo.interceptor.MyInterceptor2"/>
</mvc:interceptor>
</mvc:interceptors>
拦截器1的代码
//自定义拦截器
public class MyInterceptor1 implements HandlerInterceptor {
/*
* 预处理,controller方法执行前
* return true放行,执行下一个拦截器,如果没有,执行controller中的方法
* return false不放行
* */
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("MyInterceptor1执行了......前111(第一个拦截器)");
//request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
//true为放行,false为不放行
return true;
}
/*
* 后处理方法
* controller方法执行后
* success.jsp执行之前
* */
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("MyInterceptor1执行了......后111(第一个拦截器)");
request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
}
/*
* success.jsp页面执行后,该方法会执行
* 这个页面不能再跳转其他页面了
* */
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("MyInterceptor1执行了......最后111(第一个拦截器)");
}
}
拦截器2的代码
//自定义拦截器
public class MyInterceptor2 implements HandlerInterceptor {
/*
* 预处理,controller方法执行前
* return true放行,执行下一个拦截器,如果没有,执行controller中的方法
* return false不放行
* */
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("MyInterceptor1执行了......前222(第二个拦截器)");
//request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
//true为放行,false为不放行
return false;
}
/*
* 后处理方法
* controller方法执行后
* success.jsp执行之前
* */
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("MyInterceptor1执行了......后222(第二个拦截器)");
//request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
}
/*
* success.jsp页面执行后,该方法会执行
* 这个页面不能再跳转其他页面了
* */
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("MyInterceptor1执行了......最后222(第二个拦截器)");
}
}