前一篇博文对过滤器的定义做了说明,过滤器属于Servlet范畴的API,与Spring 没什么关系。Web开发中,我们除了使用 Filter 来过滤请web求外,还可以使用Spring提供的HandlerInterceptor(拦截器)。HandlerInterceptor 的功能跟过滤器类似,但是提供更精细的的控制能力:在request被响应之前、request被响应之后、视图渲染之前以及request全部结束之后。我们不能通过拦截器修改request内容,但是可以通过抛出异常(或者返回false)来暂停request的执行。
配置拦截器也很简单,Spring 为什么提供了基础类WebMvcConfigurerAdapter ,我们只需要重写 addInterceptors方法添加注册拦截器。
实现自定义拦截器只需要3步:
1、创建我们自己的拦截器类并实现 HandlerInterceptor 接口或继承HandlerInterceptorAdapter。
2、创建一个Java类继承WebMvcConfigurerAdapter,并重写 addInterceptors 方法。
3、实例化我们自定义的拦截器,然后将对像手动添加到拦截器链中(在addInterceptors方法中添加)。
说明:本文重点在如何在Spring-Boot中使用拦截器,关于拦截器的原理请大家查阅资料了解。
创建拦截器
==========================LoginTimeInterceptor.java============================
package com.wx.Interceptor;
import java.util.Calendar;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;
public class LoginTimeInterceptor extends HandlerInterceptorAdapter{
private int startTime;
private int endTime;
//依赖注入,请看配置文件
public void setStartTime(int startTime) {
this.startTime = startTime;
}
public void setEndTime(int endTime) {
this.endTime = endTime;
}
//在控制器执行前调用
public boolean preHandle(HttpServletRequest request,
HttpServletResponse response, Object handler) throws Exception {
System.out.println("执行preHandle方法-->01");
Calendar cal = Calendar.getInstance();
//获得当前时间对应的小时数,例如:12:05-->12,13:15-->13
int hour = cal.get(Calendar.HOUR_OF_DAY);
if (startTime <= hour && hour < endTime) {
return true; //通过拦截器,继续执行请求
} else {//给定的时间之外禁止登录
request.setAttribute("msg", "非登录时段");
request.getRequestDispatcher("Login.jsp").forward(request, response);
return false; //没有通过拦截器,返回登录页面
}
}
//在后端控制器执行后调用
public void postHandle(HttpServletRequest request,
HttpServletResponse response, Object handler,
ModelAndView modelAndView) throws Exception {
System.out.println("执行postHandle方法-->02");
super.postHandle(request, response, handler, modelAndView);
}
//整个请求执行完成后调用
public void afterCompletion(HttpServletRequest request,
HttpServletResponse response, Object handler, Exception ex)
throws Exception {
System.out.println("执行afterCompletion方法-->03");
super.afterCompletion(request, response, handler, ex);
}
}
知识讲解:此拦截器作用:如果用户没有在8-18点登录,则重定向到登录页面。注意:
(1)拦截器必须继承HandlerInterceptorAdapter类
(2)preHandle方法在后端控制器执行前被调用,postHandle方法在后端
控制器执行后被调用;afterCompletion方法在整个请求处理完成后被调用。
(3) preHandle方法:返回true,映射处理器执行链将继续执行;当返回false时,DispatcherServlet处理器认为拦截器已经处理完了请求,而不
继续执行执行链中的其它拦截器和处理器。它的API文档解释如下:true if the execution chainshould proceed with the next interceptor or the handler itself. Else,DispatcherServlet assumes that this interceptor has already dealt with theresponse itself.
(4)这三个方法都是相同的参数,Object handler参数可以转化成一个后端控制器对象,比如这里可以转换成LoginControl对象。
添加拦截器
package com.wx.boot;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;
import com.wx.Interceptor.LoginTimeInterceptor;
@Configuration
public class InterceptorConfig extends WebMvcConfigurerAdapter{
@Override
public void addInterceptors(InterceptorRegistry registry) {
//这里可以添加多个拦截器
registry.addInterceptor(new LoginTimeInterceptor()).addPathPatterns("/**");
super.addInterceptors(registry);
}
}
附,这里新增了2个全局配置
server.port=8080
server.servlet-path=*.php
spring.mvc.view.prefix=/
spring.mvc.view.suffix=.jsp
logging.level.org.springframework=INFO
#配置ContextPath
server.context-path=/BootInterceptor
#jsp,servlet修改后无需重新发布
server.jsp-servlet.init-parameters.development=true
一个是配置Context-path
Spring boot默认是/ ,这样直接通过http://ip:port/就可以访问到index页面,如果要修改为http://ip:port/path/ 访问
的话,那么需要在Application.properties文件中加入server.context-path = /你的path,比如:spring-boot,那么访问
地址就是http://ip:port/spring-boot 路径。
第二个是配置jsp,servlet修改后无需重新发布
如果修改jsp后每次都要重新发布,那开发效率就会太低了。
这个两个配置很重要!
好了,发布运行