SpringMVC(五):异常处理和拦截器

异常处理和拦截器

文章目录

    • 异常处理和拦截器
      • 1. 异常处理
        • 1.1 异常处理的流程
        • 1.2 实现springMVC全局异常处理
          • 1.2.1 环境准备
          • 1.2.2 自定义异常处理器
          • 1.2.3 在spring.xml中配置异常处理器
          • 1.2.4 测试
      • 2. 拦截器
        • 2.1 拦截器和过滤器的区别
        • 2.2 自定义拦截器
        • 2.3 多个拦截器

1. 异常处理

1.1 异常处理的流程

系统中异常包括两类:预期异常和运行时异常 RuntimeException。

​ 前者通过捕获异常从而获取异常信息,后者主要通过规范代码开发、测试通过手段减少运行时异常的发生。系统的 dao、 service、 controller 出现都通过 throws Exception 向上抛出,最后由 springmvc 前端控制器处理。这个时候有两种处理异常的方式。

  • 第一种方式

    将异常页面返回给用户,这种方式并不友好

​ 返回给用户不友好的页面

  • 第二种方式

    处理异常,将包装好的错误页面返回给用户

1.2 实现springMVC全局异常处理

1.2.1 环境准备
  • 创建maven项目,添加依赖
 <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.xml

<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>
  • 配置springmvc.xml

<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控制器代码
@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}


1.2.2 自定义异常处理器

​ 创建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;
    }
}
1.2.3 在spring.xml中配置异常处理器
<bean class="com.zmysna.utils.CustomerExceptionResolver">bean>
1.2.4 测试

​ 在浏览器上输入localhost:8080/save

2. 拦截器

​ Spring MVC 的处理器拦截器类似于 Servlet 开发中的过滤器 Filter,用于对处理器进行预处理和后处理。用户可以自己定义一些拦截器来实现特定的功能。谈到拦截器,还要向大家提一个词——拦截器链( Interceptor Chain)。拦截器链就是将拦截器按一定的顺序联结成一条链。在访问被拦截的方法或字段时,拦截器链中的拦
截器就会按其之前定义的顺序被调用。

2.1 拦截器和过滤器的区别

过滤器是 servlet 规范中的一部分, 任何 java web 工程都可以使用。

​ 拦截器是 SpringMVC 框架自己的,只有使用了 SpringMVC 框架的工程才能用。

过滤器在 url-pattern 中配置了/*之后,可以对所有要访问的资源拦截。

​ 拦截器它是只会拦截访问的控制器方法,如果访问的是 jsp, html,css,image 或者 js 是不会进行拦截的。它也是 AOP 思想的具体应用。

2.2 自定义拦截器

  • 编写controller方法,被拦截器拦截
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";
    }

}
  • 编写自定义拦截器,实现 HandlerInterceptor 接口
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方法");
    }
}
  • 在springmvc.xml中配置拦截器
<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方法

拦截器的执行流程如下图所示

2.3 多个拦截器

​ 复制拦截器HandlerInterceptorDemo1重命名为HandlerInterceptorDemo2,并在springmvc.xml中进行配置。如下面代码所示

  • HandlerInterceptorDemo2.java
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方法");
    }
}
  • springmvc.xml配置
<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方法

多个拦截器的执行流程如下

你可能感兴趣的:(java,SSM)