SpringMVC——自定义拦截器、异常处理以及父子容器配置
自定义拦截器:
一、若想实现自定义拦截器,需要实现 org.springframework.web.servlet.HandlerInterceptor 接口。
二、HandlerInterceptor API
1. 接口中定义了三个方法
2.preHandle()
(1)调用时机
在 org.springframework.web.servlet.DispatcherServlet#doDispatch 方法中。
org.springframework.web.servlet.HandlerExecutionChain#applyPreHandle 具体的调用过程
可以看到,preHander() 方法的调用是在调用目标方法前调用的。
同时可以看到,通过循环,调用了所有的拦截器,并且如果自定义的拦截器的 preHandler() 返回 false 的情况下,还是会去调用拦截器的 afterCompletion() 方法。
(2)可以获取到的资源
HttpServletRequest、HttpServletResponse、以及目标处理器。
3.postHandler()
(1)调用的时机
可以看到在调用目标方法后,渲染视图前调用的 postHandler() 方法。
具体的调用过程:
去调用每一个拦截器的 postHandler() 方法,注意循环的方式,可以看到拦截器栈 preHandler() 和 postHandler() 方法调用形成了一个 U 形。
(2)可以获取到的资源
HttpServletRequest、HttpServletResponse、目标hanler 处理器、以及返回的 ModelAndView 对象。
3.afterCompletion()
(1)调用时机
可以看到,org.springframework.web.servlet.DispatcherServlet#doDispatch 执行逻辑的过程,捕获到异常后,最终都是由拦截器的 afterCompletion() 方法进行的处理。
(2)可以获取到的资源
HttpServletReques, HttpServletResponse, 目标 handler 处理器, 所有的异常信息
4. HandlerInterceptor 家族
三、实现自定义的拦截器
1.继承 HandlerInterceptorAdapter ,按照实际需求实现对应的方法就ok。
2.在 SpringMVC Config 文件中配置。
e1:
/**
* @author solverpeng
* @create 2016-09-01-15:56
*/
public class FirstInterceptor extends HandlerInterceptorAdapter {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("preHandle");
return true;
} @Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("postHandle");
} @Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("afterCompletion");
}
}
<mvc:interceptors>
<bean class="com.nucsoft.springmvc.interceptor.FirstInterceptor"/>
</mvc:interceptors>
说明:此种配置可以拦截器所有请求。
e2:
/**
* @author solverpeng
* @create 2016-09-01-16:08
*/
public class SecondInterceptor extends HandlerInterceptorAdapter {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("SecondInterceptor#preHandle");
return true;
} @Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("SecondInterceptor#postHandle");
} @Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("SecondInterceptor#afterCompletion");
}
}
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/test/**"/>
<bean class="com.nucsoft.springmvc.interceptor.SecondInterceptor"/>
</mvc:interceptor>
</mvc:interceptors>
说明:只会拦截路径以 test 开头的请求。如:http://localhost:8080/springmvc/test/testInterceptor02 会被 SecondInterceptor 拦截。
四、拦截器的几个用处
1.性能监控
/**
* 计算每次请求耗时
* @author solverpeng
* @create 2016-09-01-17:46
*/
public class ConsumeTimeInterceptor extends HandlerInterceptorAdapter{
// SpringMVC 是单例的,所以对于每一次请求,从 ThreadLocal 中获取。
private NamedThreadLocal<Long> consumeTime = new NamedThreadLocal<>("consume_time"); @Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
consumeTime.set(System.currentTimeMillis());
System.out.println("ConsumeTimeInterceptor#preHandle");
return true;
} @Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("ConsumeTimeInterceptor#afterCompletion");
Long threadEnd = System.currentTimeMillis();
Long threadStart = consumeTime.get();
Long consume = threadEnd - threadStart;
if(consume > 10) {
System.out.println("add to log: " + consume);
}
}
}
<mvc:interceptors>
<bean class="com.nucsoft.springmvc.interceptor.ConsumeTimeInterceptor"/>
<bean class="com.nucsoft.springmvc.interceptor.FirstInterceptor"/>
</mvc:interceptors>
控制台输出:
从中也可以看出:拦截器的顺序与 SpringMVC 中配置的顺序是一致的。
2.登录检测
检测用户是否登录。
异常处理:
1.局部的
在目标 handler 类中,声明一个有 @ExceptionHandler 标注的方法,通过 @ExceptionHandler 的 value 的属性来指定需要处理的异常类型。
可以处理的异常可以是一个类型,也可以是多个类型。如:
@ExceptionHandler(value = {ArithmeticException.class}) @ExceptionHandler(value = {ArithmeticException.class, NullPointerException.class})
如:
@Controller
public class SpringTest {
@ExceptionHandler(value = {ArithmeticException.class})
public String handleArithmeticException(Exception ex) {
System.out.println("spring test class");
ex.printStackTrace();
return "error";
}
}
handleArithmeticException() 只能处理 SpringTest 控制器中出现的 ArithmeticException 类型的异常。
2.全局的
(1)基于注解
声明一个统一的异常处理类:
需要对异常处理类添加 @ControllerAdvice 注解,对异常处理方法的标注还和局部的情况一样。如:
@ControllerAdvice
public class HandlerException { @ExceptionHandler({ArithmeticException.class})
public String handlerArithmeticException(Exception ex) {
ex.printStackTrace();
return "error";
} }
handlerArithmeticException() 可以处理所有处理器发生的 ArithmeticException 类型的异常。
(2)基于xml
如:
<bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver"
id="simpleMappingExceptionResolver">
<property name="exceptionMappings">
<props>
<prop key="ArithmeticException">error</prop>
</props>
</property>
</bean>
通过属性 exceptionMappings 来指定要处理的异常,以及出现异常返回的页面。
3.优先级
基于xml>局部>全局
4.如何在异常页面得到异常信息
request.getAttribute("javax.servlet.error.exception")
5.@ResponseStatus
可以对全局的异常类标注 @ResponseStatus 注解,通过其 value 属性来指定返回的 Http状态码。是 HttpStatus 枚举类型。
来指定返回对应状态码的错误页面。
父子容器配置:
SpringMVC 层容器可以作为 Spring 容器的子容器:
即 Web 层容器可以引用业务层容器的 Bean,而业务层容器却访问不到 WEB 层容器 Bean。
SpringMVC Config:
<context:component-scan base-package="com.nucsoft.springmvc" use-default-filters="false">
<!-- 只扫描 @Controller 和 @ControllerAdvice 注解-->
<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
<context:include-filter type="annotation" expression="org.springframework.web.bind.annotation.ControllerAdvice"/>
</context:component-scan>
Spring Config:
<context:component-scan base-package="com.nucsoft.springmvc">
<!-- 不扫描 @Controller 和 @ControllerAdvice 注解-->
<context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
<context:exclude-filter type="annotation" expression="org.springframework.web.bind.annotation.ControllerAdvice"/>
</context:component-scan>
SpringMVC——自定义拦截器、异常处理以及父子容器配置的更多相关文章
- SpringMvc自定义拦截器
SpringMvc也可以使用拦截器对请求进行拦截处理,用户可以自定义拦截器来实现特定的功能,自定义拦截器必须实现HandlerInterceptor接口 -preHandle():这个方法在业务处理器 ...
- springmvc 自定义拦截器实现未登录用户的拦截
第一步:编写自定义拦截器类,该类继承HandlerInterceptorAdapter,重写preHandle方法 第二步:配置springmvc.xml文件,定义拦截器属性 登录请求的mappi ...
- SpringBoot2.X自定义拦截器实战及新旧配置对比(核心知识)
简介: 讲解拦截器使用,Spingboot2.x新版本配置拦截拦截器和旧版本SpringBoot配置拦截器区别讲解 1.@Configuration 继承WebMvcConfigurationAdap ...
- springmvc 自定义拦截器
<mvc:interceptors> <!-- 配置自定义的拦截器 --> <bean class="com.atguigu.springmvc.interce ...
- SpringMVC—Struts2拦截器学习网址整理
引自:http://blog.csdn.net/wp1603710463/article/details/49982683 SpringMVC—Struts2拦截器学习网址整理 最近项目中遇到权限相关 ...
- JavaEE开发之SpringMVC中的自定义拦截器及异常处理
上篇博客我们聊了<JavaEE开发之SpringMVC中的路由配置及参数传递详解>,本篇博客我们就聊一下自定义拦截器的实现.以及使用ModelAndView对象将Controller的值加 ...
- SpringMVC(四)-- 文件下载、自定义拦截器、异常处理
1.文件下载 用ResponseEntity<byte[]> 返回值完成文件下载 具体参见本博客之前的<文件上传下载> @RequestMapping(value=" ...
- SpringMVC 文件上传&拦截器&异常处理
文件上传 Spring MVC 为文件上传提供了直接的支持,这种支持是通过即插即用的 MultipartResolver 实现的.Spring 用 Jakarta Commons FileUpload ...
- springMVC整理04--文件上传 & 拦截器 & 异常处理
1. 文件上传 SpringMVC 的文件上传非常简便,首先导入文件上传依赖的 jar: <!-- 文件上传所依赖的 jar 包 --> <dependency> <g ...
随机推荐
- c# 动态执行脚本,相关的几个脚本引擎.
Jint 嵌入式的javascript脚本支持引擎,一直都在更新,对各种方法支持也比较好,可以 C# 交互. https://github.com/sebastienros/jint Jurass ...
- 分享一个异步任务在遇到IO异常时支持递归回调的辅助方法
public void TryAsyncActionRecursively<TAsyncResult>( string asyncActionName, Func<Task<T ...
- Aoite 系列(04) - 强劲的 CommandModel 开发模式(上篇)
Aoite 是一个适于任何 .Net Framework 4.0+ 项目的快速开发整体解决方案.Aoite.CommandModel 是一种开发模式,我把它成为"命令模型",这是一 ...
- 每周一书-《Bootstrap基础教程》
首先说明,本周活动有效时间为8月15日到21日.本周为大家送出的书是有电子工业出版,贺臣/陈鹏编著的<Bootsrap基础教程>,为前端入门必读书籍. 下面是从书中摘录的内容. “ Boo ...
- 【T-SQL基础】01.单表查询-几道sql查询题
概述: 本系列[T-SQL基础]主要是针对T-SQL基础的总结. [T-SQL基础]01.单表查询-几道sql查询题 [T-SQL基础]02.联接查询 [T-SQL基础]03.子查询 [T-SQL基础 ...
- [体感游戏] 1、MPU6050数据采集传输与可视化
最近在研究体感游戏,到目前为止实现了基于51单片机的MPU6050数据采集.利用蓝牙模块将数据传输到上位机,并利用C#自制串口数据高速采集软件,并且将数据通过自制的折线图绘制模块可视化地展示出来等功能 ...
- Unity3D使用经验总结 编辑器扩展篇
一个引擎,最重要的就是工具,工具除了提升开发速度,提供可视化操作环境以外,还带了容错功能. 它使得大家的工作局限在一定的范围内,比如一个变量的配置,或者是一些类型的选择. 使用编辑器,使得既使不太明白 ...
- 有强迫症的我只能自己写一个json格式化工具
缘由 为什么博客园的markdown解析出问题了啊?好奇怪啊! 一直以来在编码规范界有2大争论不休的话题,一个是关于是用空格缩进还是tab缩进的问题,一个是花括号是否换行的问题,笔者是tab缩进和花括 ...
- Java基础之网络编程
网络编程:1.网络编程概述 (1)网络模型 OSI参考模型 TCP/IP参考模型 (2)网络通讯要素 IP地址 端口号 传输协议 (3)网络通讯前提: **找到对方IP **数据要发送到指定端口.为了 ...
- js笔记——js里的null和undefined
以下内容摘录自阮一峰的<语法概述 -- JavaScript 标准参考教程(alpha)>章节『5.null和undefined』,以做备忘. null与undefined都可以表示&qu ...