SpringMVC基础(4):拦截器

目录

1、拦截器

2、第一个拦截器

测试postHandle和afterCompletion

3、多个拦截器

4、验证登陆案例

5、拦截器和过滤器的对比

1、拦截器

拦截器:springmvc框架中的一种对象, 需要实现接口HandlerInterceptor

拦截器的定义:

  • 创建类实现拦截器接口HandlerInterceptor,实现接口中的方法(3个)。
  • 在springmvc配置文件中,声明拦截器对象,并指定拦截的uri地址

作用: 拦截用户的请求,拦截到controller的请求。可以预先对请求做处理,根据处理结果决定是否执行controller 。 也可以把多个controller中共用的功能定义到拦截器。

特点:

  • 拦截器可以分为系统拦截器和自定义拦截器。
  • 一个项目可以多个拦截器,0或多个自定义拦截器。
  • 拦截器侧重拦截用户的请求。
  • 拦截器是在请求处理之前先执行的。

2、第一个拦截器

/**
 * 拦截器
 */
public class MyInterceptor implements HandlerInterceptor {
    /**
     * preHandle: 预先处理请求的方法。
     *  参数:
     *     Object handler:被拦截的控制器对象(即MyController)
     *  返回值: boolean
     *    true: 请求是正确的,可以被controller处理的。
     *       =====MyInterceptor拦截器的preHandle====
     *       执行了MyController的doSome方法
     *       =====MyInterceptor拦截器的postHandle====
     *       =====MyInterceptor拦截器的afterCompletion====
     *    false: 请求不能被处理, 控制器方法不会执行。 请求到此截止。
     *       =====MyInterceptor拦截器的preHandle====
     * 特点:
     *  1. 预处理方法他的执行时间: 在控制器方法之前先执行的。
     *  2. 可以对请求做处理, 可以做登录的检查, 权限的判断, 统计数据等等。
     *  3. 决定请求是否执行。
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("=====MyInterceptor拦截器的preHandle====");
        return true;
    }

    /**
     * postHandle: 后处理方法
     * 参数:
     *  Object handler : 被拦截的控制器对象(MyController)
     *  ModelAndView mv: 控制器方法的返回值(请求的执行结果)
     * 特点:
     *  1. 在控制器方法之后执行的。
     *  2. 能获取到控制器方法的执行结果,可以修改原来的执行结果。
     *     可以修改数据, 也可以修改视图
     *  3. 可以做对请求的二次处理。
     */
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView mv) throws Exception {
        System.out.println("=====MyInterceptor拦截器的postHandle====");
    }

    /**
     * afterCompletion: 最后执行的方法
     * 参数:
     *   Object handler : 被拦截的控制器对象(MyController)
     *   Exception ex: 异常对象
     * 特点:
     *  1. 在请求处理完成后执行的,请求处理完成的标志是 视图处理完成,对视图执行forward操作后。
     *  2. 可以做程序最后要做的工作: 释放内存,清理临时变量。
     *  3. 方法的执行条件:
     *     1)当前的拦截器他的preHandle()方法必须执行。
     *     2)preHandle()必须返回true。
     */
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("=====MyInterceptor拦截器的afterCompletion====");
    }
}

配置文件springmvc.xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:mvc="http://www.springframework.org/schema/mvc"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
            http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
            http://www.springframework.org/schema/context
            http://www.springframework.org/schema/context/spring-context-3.2.xsd
            http://www.springframework.org/schema/mvc
            http://www.springframework.org/schema/mvc/spring-mvc-3.2.xsd">

<!-- SpringMVC配置文件 声明controller,视图解析器等web-->
    <context:component-scan base-package="com.crane.springmvc.controller"/>

    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix" value="/WEB-INF/view/"></property>
        <property name="suffix" value=".jsp"></property>
    </bean>

    <mvc:annotation-driven/>
    <!-- 异常处理-->
    <context:component-scan base-package="com.crane.springmvc.handle"/>

    <!-- 声明拦截器 -->
    <mvc:interceptors>
        <!-- 声明第一个拦截器 -->
        <mvc:interceptor>
            <!-- 指定拦截器的拦截地址
                path: 拦截的uri 地址,使用 ** 通配符
                     例如: path="/user/**"
                       http://localhost:8080/user/listUser.do
                       http://localhost:808/user/query/queryUser.do
                下面表示拦截所有请求,请求都经过该拦截器-->
            <mvc:mapping path="/**"/>
            <bean class="com.crane.springmvc.interceptor.MyInterceptor"></bean>
        </mvc:interceptor>
    </mvc:interceptors>

</beans>

当你的preHandle返回true, 执行结果:

=====MyInterceptor拦截器的preHandle====
执行MyController的doSome方法
=====MyInterceptor拦截器的postHandle====
=====MyInterceptor拦截器的afterCompletion====

  • 请求的执行顺序: 用户some.do---preHandle---doSome---postHandle--afterComplietion

当 preHandle返回 false

=====MyInterceptor拦截器的preHandle====

测试postHandle和afterCompletion

@Controller
public class MyController {

    @RequestMapping(value="/some.do")
    public ModelAndView doSome(HttpSession session, String name, Integer age){
        System.out.println("执行MyController的doSome方法");

        //添加一个临时数据
        session.setAttribute("attr","在controller中增加的临时数据");

        ModelAndView modelAndView = new ModelAndView();
        modelAndView.addObject("myname", name);
        modelAndView.addObject("myage", age);

        modelAndView.setViewName("show");

        return modelAndView;
    }
}
/**
 * 拦截器
 */
public class MyInterceptor implements HandlerInterceptor {
    /**
     * preHandle: 预先处理请求的方法。
     *  参数:
     *     Object handler:被拦截的控制器对象(即MyController)
     *  返回值: boolean
     *    true: 请求是正确的,可以被controller处理的。
     *       =====MyInterceptor拦截器的preHandle====
     *       执行了MyController的doSome方法
     *       =====MyInterceptor拦截器的postHandle====
     *       =====MyInterceptor拦截器的afterCompletion====
     *    false: 请求不能被处理, 控制器方法不会执行。 请求到此截止。
     *       =====MyInterceptor拦截器的preHandle====
     * 特点:
     *  1. 预处理方法他的执行时间: 在控制器方法之前先执行的。
     *  2. 可以对请求做处理, 可以做登录的检查, 权限的判断, 统计数据等等。
     *  3. 决定请求是否执行。
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("=====MyInterceptor拦截器的preHandle====");
        return true;

//        request.getRequestDispatcher("tips.jsp").forward(request,response);
//            return false;
    }

    /**
     * postHandle: 后处理方法
     * 参数:
     *  Object handler : 被拦截的控制器对象(MyController)
     *  ModelAndView mv: 控制器方法的返回值(请求的执行结果)
     * 特点:
     *  1. 在控制器方法之后执行的。
     *  2. 能获取到控制器方法的执行结果,可以修改原来的执行结果。
     *     可以修改数据, 也可以修改视图
     *  3. 可以做对请求的二次处理。
     */
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView mv) throws Exception {
        System.out.println("=====MyInterceptor拦截器的postHandle====");
        //对请求二次处理
        if(mv != null){
            //修改数据
            mv.addObject("mydate", new Date());
            //修改视图
            mv.setViewName("other");
        }
    }

    /**
     * afterCompletion: 最后执行的方法
     * 参数:
     *   Object handler : 被拦截的控制器对象(MyController)
     *   Exception ex: 异常对象
     * 特点:
     *  1. 在请求处理完成后执行的,请求处理完成的标志是 视图处理完成,对视图执行forward操作后。
     *  2. 可以做程序最后要做的工作: 释放内存,清理临时变量。
     *  3. 方法的执行条件:
     *     1)当前的拦截器他的preHandle()方法必须执行。
     *     2)preHandle()必须返回true。
     */
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("=====MyInterceptor拦截器的afterCompletion====");
        //获取数据
        HttpSession session = request.getSession();
        Object attr = session.getAttribute("attr");
        System.out.println("attr"+ attr);

        //删除数据
        session.removeAttribute("attr");
        //确定数据是否删除
        attr = session.getAttribute("attr");
        System.out.println("删除后在检查数据" + attr);
    }
}

3、多个拦截器

使用两个拦截器, 主要看拦截器的执行顺序,以及哪个方法控制请求的执行

  • 1)两个拦截器,第一个preHandle=true, 第二个拦截器preHandle=true
/**
 * 拦截器
 */
public class MyInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("=====MyInterceptor1111拦截器的preHandle====");
        return true;
    }

    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView mv) throws Exception {
        System.out.println("=====MyInterceptor1111拦截器的postHandle====");
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("=====MyInterceptor1111拦截器的afterCompletion====");
    }
}

/**
 * 拦截器
 */
public class MyInterceptor2 implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("=====MyInterceptor222拦截器的preHandle====");
        return true;
    }

    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView mv) throws Exception {
        System.out.println("=====MyInterceptor222拦截器的postHandle====");
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("=====MyInterceptor222拦截器的afterCompletion====");
    }
}
<!-- 声明拦截器 -->
    <mvc:interceptors>
        <!-- 声明第一个拦截器 -->
        <mvc:interceptor>
            <!-- 指定拦截器的拦截地址
                path: 拦截的uri 地址,使用 ** 通配符
                     例如: path="/user/**"
                       http://localhost:8080/user/listUser.do
                       http://localhost:808/user/query/queryUser.do
                下面表示拦截所有请求,请求都经过该拦截器-->
            <mvc:mapping path="/**"/>
            <bean class="com.crane.springmvc.interceptor.MyInterceptor"></bean>
        </mvc:interceptor>

        <!-- 声明第二个拦截器 -->
        <mvc:interceptor>
            <!-- 指定拦截器的拦截地址
                path: 拦截的uri 地址,使用 ** 通配符
                     例如: path="/user/**"
                       http://localhost:8080/user/listUser.do
                       http://localhost:808/user/query/queryUser.do
                下面表示拦截所有请求,请求都经过该拦截器-->
            <mvc:mapping path="/**"/>
            <bean class="com.crane.springmvc.interceptor.MyInterceptor2"></bean>
        </mvc:interceptor>

    </mvc:interceptors>

=====MyInterceptor1111拦截器的preHandle====
=====MyInterceptor2222拦截器的preHandle====
执行MyController的doSome方法
=====MyInterceptor2222拦截器的postHandle====
=====MyInterceptor1111拦截器的postHandle====
=====MyInterceptor2222拦截器的afterCompletion====
=====MyInterceptor1111拦截器的afterCompletion====

请求的执行顺序:

用户some.do—拦截器1的preHandle----拦截器2preHandle—控制器doSome—拦截器2postHandle—拦截器1的postHandle—拦截器2的afterCompletion—拦截器1的afterCompletion。

  • 2)两个拦截器,第一个preHandle=true, 第二个拦截器preHandle=false
     

=====MyInterceptor1111拦截器的preHandle====
=====MyInterceptor2222拦截器的preHandle====
=====MyInterceptor1111拦截器的afterCompletion====

  • 3)两个拦截器,第一个preHandle=false, 第二个拦截器preHandle=true|false

=====MyInterceptor1111拦截器的preHandle====

为什么要使用多个拦截器 ?

  • 1、把验证功能分散到独立的拦截器, 每个拦截器做单一的验证处理:验证登陆和权限
  • 2、组合多个拦截器。

总结:

多个拦截器, 串在一个链条上的。 多个拦截器和一个控制器对象在一个链条上 ,框架中使用HandlerExecutionChain(处理器执行链),表示这个执行链条: 

public class HandlerExecutionChain {
    private final Object handler;
    // 存放控制器对象的,MyController
     @Nullable
    private HandlerInterceptor[] interceptors;
   // 存放多个拦截器对象的。MyInterceptor 1, 2
   @Nullable
   private List<HandlerInterceptor> interceptorList;

}

拦截器怎么实现 1,2, 2,1的执行顺序, 遍历HandlerInterceptor[] interceptors 数组

HandlerInterceptor[] interceptors = {MyInterceptor1, MyInterceptor2};
    //循环调用方法 1-2
    for(int i=0;i<interceptors.length;i++){
          HandlerInterceptor  obj= interceptors[i];
          obj.preHandle();
    }
    MyController.doSome();
    // 2-1
    for(int i=interceptors.length-1 ;i>=0;i--){
        HandlerInterceptor  obj= interceptors[i];
        obj.postHandle();
    }

4、验证登陆案例

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>SSM</title>
</head>

<body>
<h3>登陆</h3>
<%  session.setAttribute("username","zhangsan");%>
</body>
</html>

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>SSM</title>
</head>

<body>
<h3>退出</h3>
<%  session.removeAttribute("username") ;%>
</body>
</html>

/**
 * 拦截器
 */
public class MyInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("=====MyInterceptor1111拦截器的preHandle====");
        //获取登陆信息
        Object attr = request.getAttribute("username");
        String username = "";
        if(username != null){
            username = (String)attr;
        }

        if("zhangsan".equals(username)){
            return true;
        }
        else{
            request.getRequestDispatcher("tips.jsp").forward(request,response);
            return false;
        }
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView mv) throws Exception {
        System.out.println("=====MyInterceptor1111拦截器的postHandle====");
    }

   
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("=====MyInterceptor1111拦截器的afterCompletion====");
    }
}

5、拦截器和过滤器的对比

  1. 拦截器是springmvc框架中的对象; 过滤器是servlet中的对象。
  2. 拦截器对象是框架容器创建的;过滤器对象是tomcat创建的对象。
  3. 拦截器是侧重对请求做判断的,处理的,可以截断请求。 过滤器是侧重对request,response对象的属性,参数设置值的。 例如request.setCharacterEncoding(“utf-8”)。
  4. 拦截器的执行时间有三个:控制器方法之前、之后、请求完成后。 过滤器是在请求之前。
  5. 拦截器是拦截对controller动态资源请求的。 过滤器可以过滤所有请求:动态的和静态的。
  6. 拦截器和过滤器一起执行的,先执行的过滤器,后面是 中央调度器,后面才是拦截器,再后面是控制器方法。 

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值