我的springMVC+mybatis中的interceptor使用@autowired注入DAO失败,导致报空指针错误,这个是为什么呢?

:空指针说明没有注入进来,你可以检查一下你的这个拦截器interceptor是不是加了注解@component,并且spring配置文件扫描了这个类,你的类不交给spring容器管理,就不提注入这回事了。

一、Servlet Filter与Spring interceptor的执行顺序

Filter有顺序吗?我们怎么控制filter的执行顺序。通过Tomcat的代码分析,servlet在Filter执行完成后才调用,如有多个filter怎么控制执行顺序,首先会想到在web.xml配置某个参数,例如order之类的,但查找一下一番,servlet并没有这个参数。试试filter Mapping的配置的先后顺序,果然有效,原来filter的执行顺序就考filter mapping在web.xml中的顺序。

spring interceptor也是这样的执行顺序,不过interceptor多一个配置参数order通过他也可以来实现interceptor的执行顺序。很多应用场景中,执行顺序还是重要的,比如cache和transaction interceptor的执行顺序,很显然cache应该在transaction之前,这样发现命中了就不用打开事务,如果transaction在前,每次都打开事务即使cache命中,这是一个无谓东动作。

二、利用springMVC的interceptor实现页面性能监控(Filter亦可)

调优第一步,找出耗时比较长的页面进行优化。利用interceptor能轻易搞定。interceptor提供了preHandle和postHandle以及afterCompletion三个方法。preHandle调用controller具体方法之前调用,postHandle完成具体方法之后调用,afterCompletion完成对页面的render以后调用,至此整个页面渲染完成。也就是说我们在preHandle记录开始的时间,在afterCompletion记录结束的时间,就可或者整个页面生成的时间。Spring自带StopWatch工具类来实现时间跟踪,关键一点interceptor不是线程安全的。我们需要借助threadlocal来实现线程安全。

  1. @Override
  2. public boolean preHandle(HttpServletRequest request,
  3. HttpServletResponse response, Object handler) throws Exception {
  4. if(usePerformance){
  5. StopWatch stopWatch = new StopWatch(handler.toString());
  6. stopWatchLocal.set(stopWatch);
  7. stopWatch.start(handler.toString());
  8. }
  9. return true;
  10. }
  11. @Override
  12. public void afterCompletion(HttpServletRequest request,
  13. HttpServletResponse response, Object handler, Exception ex)
  14. throws Exception {
  15. if(usePerformance){
  16. StopWatch stopWatch = stopWatchLocal.get();
  17. stopWatch.stop();
  18. String currentPath = request.getRequestURI();
  19. String queryString  = request.getQueryString();
  20. queryString = queryString == null ? "":"?" + queryString;
  21. log.info("access url path:" + currentPath + queryString +  " |time:" + stopWatch.getTotalTimeMillis());
  22. stopWatchLocal.set(null);
  23. }
  24. }

三、SpringMVC 拦截器实现分析

SpringMVC的拦截器不同于Spring的拦截器,SpringMVC具有统一的入口DispatcherServlet,所有的请求都通过DispatcherServlet,所以只需要在DispatcherServlet上做文章即可,DispatcherServlet也没有代理,同时SpringMVC管理的Controller也不有代理。哪不难想到我们在执行controller之前做某些动作,执行完毕做某些动作,render完成做某些动作。SpringMVC的拦截器对应提供了三个preHandle,postHandle,afterCompletion方法。只需在三个方法内写我们需要的逻辑就行,多了都是废话,还是代码实在。

如果你没有使用springMVC可以使用filter来完成:

  1. stopWatch.start();
  2. doFilterChain();
  3. stopWatch.stop();

三、SpringMVC 拦截器实现分析

SpringMVC的拦截器不同于Spring的拦截器,SpringMVC具有统一的入口DispatcherServlet,所有的请求都通过DispatcherServlet,所以只需要在DispatcherServlet上做文章即可,DispatcherServlet也没有代理,同时SpringMVC管理的Controller也不有代理。哪不难想到我们在执行controller之前做某些动作,执行完毕做某些动作,render完成做某些动作。SpringMVC的拦截器对应提供了三个preHandle,postHandle,afterCompletion方法。只需在三个方法内写我们需要的逻辑就行,多了都是废话,还是代码实在。

  1. HandlerInterceptor[] interceptors = mappedHandler.getInterceptors();
  2. if (interceptors != null) {
  3. for (int i = 0; i < interceptors.length; i++) {
  4. HandlerInterceptor interceptor = interceptors[i];
  5. //ha.handle是调用具体的controller在此之前执行preHandle                      if (!interceptor.preHandle(processedRequest, response, mappedHandler.getHandler())) {
  6. triggerAfterCompletion(mappedHandler, interceptorIndex, processedRequest, response, null);
  7. return;
  8. }
  9. interceptorIndex = i;
  10. }
  11. }
  12. // Actually invoke the handler.
  13. mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

完成调用之后,调用render(),最后执行afterCompletion()。

  1. if (interceptors != null) {
  2. for (int i = interceptors.length - 1; i >= 0; i--) {
  3. HandlerInterceptor interceptor = interceptors[i];
  4. interceptor.postHandle(processedRequest, response, mappedHandler.getHandler(), mv);
  5. }
  6. }
  7. }
  8. catch (ModelAndViewDefiningException ex) {
  9. logger.debug("ModelAndViewDefiningException encountered", ex);
  10. mv = ex.getModelAndView();
  11. }
  12. catch (Exception ex) {
  13. Object handler = (mappedHandler != null ? mappedHandler.getHandler() : null);
  14. mv = processHandlerException(processedRequest, response, handler, ex);
  15. errorView = (mv != null);
  16. }
  17. // Did the handler return a view to render?
  18. if (mv != null && !mv.wasCleared()) {
  19. render(mv, processedRequest, response);
  20. if (errorView) {
  21. WebUtils.clearErrorRequestAttributes(request);
  22. }
  23. }
  24. else {
  25. if (logger.isDebugEnabled()) {
  26. logger.debug("Null ModelAndView returned to DispatcherServlet with name '" + getServletName() +
  27. "': assuming HandlerAdapter completed request handling");
  28. }
  29. }
  30. // Trigger after-completion for successful outcome.
  31. triggerAfterCompletion(mappedHandler, interceptorIndex, processedRequest, response, null);

spring mvc拦截器原理分析的更多相关文章

  1. SSM(spring mvc+spring+mybatis)学习路径——2-2、spring MVC拦截器

    目录 2-2 Spring MVC拦截器 第一章 概述 第二章 Spring mvc拦截器的实现 2-1 拦截器的工作原理 2-2 拦截器的实现 2-3 拦截器的方法介绍 2-4 多个拦截器应用 2- ...

  2. spring mvc 拦截器的使用

    Spring MVC 拦截器的使用 拦截器简介 Spring MVC 中的拦截器(Interceptor)类似于 Servler 中的过滤器(Filter).用于对处理器进行预处理和后处理.常用于日志 ...

  3. 【Java Web开发学习】Spring MVC 拦截器HandlerInterceptor

    [Java Web开发学习]Spring MVC 拦截器HandlerInterceptor 转载:https://www.cnblogs.com/yangchongxing/p/9324119.ht ...

  4. Spring Boot 2.X(九):Spring MVC - 拦截器(Interceptor)

    拦截器 1.简介 Spring MVC 中的拦截器(Interceptor)类似于 Servlet 开发中的过滤器 Filter,它主要用于拦截用户请求并作相应的处理,它也是 AOP 编程思想的体现, ...

  5. Spring MVC拦截器配置

    Spring MVC拦截器配置 (1)自定义拦截器 package learnspringboot.xiao.other; import org.springframework.web.servlet ...

  6. 写的太细了!Spring MVC拦截器的应用,建议收藏再看!

    Spring MVC拦截器 拦截器是Spring MVC中强大的控件,它可以在进入处理器之前做一些操作,或者在处理器完成后进行操作,甚至是在渲染视图后进行操作. 拦截器概述 对于任何优秀的MVC框架, ...

  7. 对于Spring MVC 拦截器的一些了解

    Spring MVC 拦截器的执行顺序 应用场景 假设请求 localhost:8080/ 则要求直接重定向到 localhost:8080/login ; 定义拦截器顺序 permission lo ...

  8. Spring MVC拦截器浅析

    Spring MVC拦截器 重点:Spring MVC的拦截器只会拦截控制器的请求,如果是jsp.js.image.html则会放行. 什么是拦截器 运行在服务器的程序,先于Servlet或JSP之前 ...

  9. spring mvc拦截器

    Java里的拦截器是动态拦截Action调用的对象.它提供了一种机制可以使开发者可以定义在一个action执行的前后执行的代码,也可以在一个action执行前阻止其执行,同时也提供了一种可以提取act ...

随机推荐

  1. C++模板 · 为什么要引入模板机制?

    刚学过类模板时,很不理解,甚至觉得这简直没有用,在自己骗自己嘛!明明很方便的东西,偏偏要加个类模板来回折腾.可能因为我们刚开始写的程序很简单,有时候,可能程序复杂一点,对理解一些概念更有帮助. 今天在 ...

  2. heartbeat+drdb+nfs实现高可用

    一.环境 nfsserver01:192.168.127.101 心跳:192.168.42.101 centos7.3 nfsserver02:192.168.127.102 心跳:192.168. ...

  3. 编译openwrt_MT7688_hiwooya

    参考链接: 无涯论坛地址: http://www.hi-wooya.com/forum.php openwrt官网地址:https://openwrt.org/zh-cn/doc/howto/buil ...

  4. vs实用插件

    Live Share 强烈推荐的一款插件,能在VS程序中打开文件并且显示他的效果.非常非常实用!,具体功能介绍在你搜索该插件时候有说明,非常非常好用的一款插件! 后续插件推荐转载参考与其他博主 1.C ...

  5. 创建虚拟机、安装centos系统,xshell连接虚拟机

    创建虚拟机 文件--->新建虚拟机--->自定义最高级 选择虚拟机兼容性: workstation12.0:限制少,而且兼容的多 稍后安装操作系统 选择Linux系统 可选择centos7 ...

  6. CodeForces 149D 区间DP Coloring Brackets

    染色有三个条件: 对于每个点来说要么不染色,要么染红色,要么染蓝色 对于每对配对的括号来说,有且只有一个一边的括号被染色 相邻的括号不能染成相同的颜色 首先可以根据给出的括号序列计算出括号的配对情况, ...

  7. Java-获取一个类的父类

    如何使用代码获取一个类的父类 package com.tj; public class MyClass implements Cloneable { public static void main(S ...

  8. [转]python 多线程threading简单分析

    多线程和多进程是什么自行google补脑 对于python 多线程的理解,我花了很长时间,搜索的大部份文章都不够通俗易懂.所以,这里力图用简单的例子,让你对多线程有个初步的认识. 单线程 在好些年前的 ...

  9. jmeter历史版本下载

    https://archive.apache.org/dist/jmeter/binaries/

  10. Dell Omsa在Linux服务器上安装部署

    前言 本页详述了在一台Linux(RHEL6.4 x86_64)服务器上部署安装OMSA的通用做法,包括OMSA软件的获取方法和安装步骤. 演示环境: PowerEdge R620, RHEL 6.4 ...