系统中异常包括两类:预期异常和运行时异常 RuntimeException。
前者通过捕获异常从而获取异常信息,后者主要通过规范代码开发、测试通过手段减少运行时异常的发生。系统的 dao、 service、 controller 出现都通过 throws Exception 向上抛出,最后由 springmvc 前端控制器处理。这个时候有两种处理异常的方式。
第一种方式
将异常页面返回给用户,这种方式并不友好
返回给用户不友好的页面
第二种方式
处理异常,将包装好的错误页面返回给用户
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>jsp-api</artifactId>
<version>2.0</version>
</dependency>
<dependency>
<groupId>log4j</groupId>
<artifactId>log4j</artifactId>
<version>1.2.17</version>
</dependency>
</dependencies>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://java.sun.com/xml/ns/javaee"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
version="2.5">
<servlet>
<servlet-name>dispatcherServletservlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServletservlet-class>
<init-param>
<param-name>contextConfigLocationparam-name>
<param-value>classpath:springmvc.xmlparam-value>
init-param>
servlet>
<servlet-mapping>
<servlet-name>dispatcherServletservlet-name>
<url-pattern>/url-pattern>
servlet-mapping>
web-app>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:mvc="http://www.springframework.org/schema/mvc"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd">
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/pages/">property>
<property name="suffix" value=".jsp">property>
bean>
<mvc:annotation-driven/>
<context:component-scan base-package="com.zmysna.controller">context:component-scan>
beans>
@Controller
public class AccountController {
/**
* 模拟异常
*/
@RequestMapping("/save")
public String save() {
//除零异常
int i = 1 / 0;
return "success";
}
}
error.jsp错误页面
包装好的错误页面。这里简单回显一些给用户的信息,可以进一步的包装
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Error
${error}
创建java类实现HandlerExceptionResolver接口
package com.zmysna.utils;
import org.springframework.web.servlet.HandlerExceptionResolver;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class CustomerExceptionResolver implements HandlerExceptionResolver {
@Override
public ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) {
//打印异常信息
System.out.println(e);
ModelAndView modelAndView = new ModelAndView();
//设置跳转的逻辑视图名,可以由视图解析器解析成实际URL
modelAndView.setViewName("error");
//往request域添加数据
modelAndView.addObject("error","你的页面不见了");
return modelAndView;
}
}
<bean class="com.zmysna.utils.CustomerExceptionResolver">bean>
在浏览器上输入localhost:8080/save
Spring MVC 的处理器拦截器类似于 Servlet 开发中的过滤器 Filter,用于对处理器进行预处理和后处理。用户可以自己定义一些拦截器来实现特定的功能。谈到拦截器,还要向大家提一个词——拦截器链( Interceptor Chain)。拦截器链就是将拦截器按一定的顺序联结成一条链。在访问被拦截的方法或字段时,拦截器链中的拦
截器就会按其之前定义的顺序被调用。
过滤器是 servlet 规范中的一部分, 任何 java web 工程都可以使用。
拦截器是 SpringMVC 框架自己的,只有使用了 SpringMVC 框架的工程才能用。
过滤器在 url-pattern 中配置了/*之后,可以对所有要访问的资源拦截。
拦截器它是只会拦截访问的控制器方法,如果访问的是 jsp, html,css,image 或者 js 是不会进行拦截的。它也是 AOP 思想的具体应用。
package com.zmysna.controller;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
public class DemoController {
public DemoController() {
System.out.println("2.创建demoController对象");
}
@RequestMapping("testInterrupt")
public String testInterrupt() {
System.out.println("4.执行控制器testInterrupt()方法");
return "success";
}
}
package com.zmysna.intercept;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class HandlerInterceptorDemo1 implements HandlerInterceptor {
public HandlerInterceptorDemo1(){
System.out.println("1. 创建HandlerInterceptorDemo1对象");
}
/**
* 拦截的方法,业务逻辑控制校验代码都写到这里。
* 此方法返回 true ,表示放行。放行的含义是指,如果有下一个拦截器就执行下一个,
* 如果该拦截器处于拦截器链的最后一个,则执行控制器中的方法。
* 此方法返回 false ,表示不放心,不会执行下一个拦截器以及控制器
*
* 如何调用? 按拦截器定义顺序调用
* 何时调用? 只要配置了都会调用
* 有什么用? 如果程序员决定该拦截器对请求进行拦截处理后还要调用其他的拦截器,
* 或者是业务处理器去进行处理,则返回 true 。
* 如果程序员决定不需要再调用其他的组件去处理请求,则返回 false 。
*/
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("3. 执行HandlerInterceptorDemo1的preHandle方法");
return true;
}
/**
* 如何调用? 按拦截器定义逆序调用
* 何时调用? 在拦截器链内所有拦截器返成功调用
* 有什么用? 在业务处理器处理完请求后,但是 DispatcherServlet 向客户端返回响应前被调用。
*/
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("5. 执行HandlerInterceptorDemo1的postHandle方法");
}
/**
* 如何调用?按拦截器定义逆序调用
* 何时调用?只有 preHandle 返回 true 才调用
* 有什么用?在 DispatcherServlet 完全处理完请求后被调用,可以在该方法中进行一些资源清理的操作。
*/
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("6. 执行HandlerInterceptorDemo1的afterCompletion方法");
}
}
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/**">mvc:mapping>
<bean class="com.zmysna.intercept.HandlerInterceptorDemo1">bean>
mvc:interceptor>
mvc:interceptors>
测试
使用浏览器进行访问localhost:8080/testInterrupt,控制台结果如下
1. 创建HandlerInterceptorDemo1
2.创建demoController对象
3. 执行HandlerInterceptorDemo1的preHandle方法
4.执行控制器testInterrupt()方法
5. 执行HandlerInterceptorDemo2的postHandle方法
6. 执行HandlerInterceptorDemo2的afterCompletion方法
拦截器的执行流程如下图所示
复制拦截器HandlerInterceptorDemo1重命名为HandlerInterceptorDemo2,并在springmvc.xml中进行配置。如下面代码所示
public class HandlerInterceptorDemo2 implements HandlerInterceptor {
public HandlerInterceptorDemo2(){
System.out.println("1. 创建HandlerInterceptorDemo2对象");
}
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("3. 执行HandlerInterceptorDemo2的preHandle方法");
return true;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("5. 执行HandlerInterceptorDemo2的postHandle方法");
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("6. 执行HandlerInterceptorDemo2的afterCompletion方法");
}
}
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/**">mvc:mapping>
<bean class="com.zmysna.intercept.HandlerInterceptorDemo1">bean>
mvc:interceptor>
<mvc:interceptor>
<mvc:mapping path="/**">mvc:mapping>
<bean class="com.zmysna.intercept.HandlerInterceptorDemo2">bean>
mvc:interceptor>
mvc:interceptors>
1. 创建HandlerInterceptorDemo1对象
1. 创建HandlerInterceptorDemo2对象
2.创建demoController对象
3. 执行HandlerInterceptorDemo1的preHandle方法
3. 执行HandlerInterceptorDemo2的preHandle方法
4.执行控制器testInterrupt()方法
5. 执行HandlerInterceptorDemo2的postHandle方法
5. 执行HandlerInterceptorDemo1的postHandle方法
6. 执行HandlerInterceptorDemo2的afterCompletion方法
6. 执行HandlerInterceptorDemo1的afterCompletion方法
多个拦截器的执行流程如下