过滤器是什么?
它是基于Servlet 技术实现的, 简单的来说,过滤器就是起到过滤的作用,在web项目开发中帮我们过滤一些指定的 url做一些特殊的处理。 过滤器主要做什么?
过滤器的代码实现
过滤器(filter)有三个方法,其中初始化(init)和摧毁(destroy)方法一般不会用到,主要用到的是doFilter这个方法。
怎么过滤呢?
如果过滤通过,则在doFilter 执行 filterChain.doFilter(request,response); 。


那么在springBoot中如何使用过滤器呢?
自定义Filter有两种实现方式,第一种是使用@WebFilter,第二种是使用 FilterRegistrationBean,下面我们分别来实现
@WebFilter 用于将一个类声明为过滤器,该注解将会在部署时被容器处理,容器将根据具体的属性配置将相应的类部署为过滤器。
| 属性名 | 类型 | 描述 |
|---|---|---|
| filterName | String | 指定该Filter的名称 |
| urlPatterns | String | 指定该Filter所拦截的URL。 |
| value | String | 与 urlPatterns 一致 |
package com.feng.springboot_filter.filter;
import org.springframework.core.annotation.Order;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
@WebFilter(urlPatterns = "/api/*", filterName = "myFilter")
@Order(1) //指定过滤器的执行顺序,值越大越靠后执行
public class MyFilter implements Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {
System.out.println("初始化过滤器");
Filter.super.init(filterConfig);
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) servletRequest;
String requestURI = request.getRequestURI();
String method = request.getMethod();
System.out.println("拦截器 MyFilter 拦截了请求:" + requestURI + ",方法为:" + method);
filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() {
Filter.super.destroy();
}
}

package com.feng.springboot_filter.controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/api")
public class FilterController {
@GetMapping("/user/filter")
public String hello(){
return "已经通过了过滤器";
}
}
package com.feng.springboot_filter.controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class TestController {
@GetMapping("/test")
public String hello(){
return "hello world";
}
}

http://localhost:8080/api/user/filter


http://localhost:8080/test

没有走过滤器

package com.feng.springboot_filter.config;
import com.feng.springboot_filter.filter.MyFilter;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class FilterConfig {
@Bean
public MyFilter myFilter() {
return new MyFilter();
}
@Bean
public FilterRegistrationBean getFilterRegistrationBean(MyFilter myFilter) {
FilterRegistrationBean filterRegistrationBean = new FilterRegistrationBean();
/**
* 设置过滤器
*/
filterRegistrationBean.setFilter(myFilter());
/**
* 拦截路径
*/
filterRegistrationBean.addUrlPatterns("/api/*");
/**
* 设置名称
*/
filterRegistrationBean.setName("myFilter");
/**
* 设置访问优先级 值越小越高
*/
filterRegistrationBean.setOrder(1);
return filterRegistrationBean;
}
}
//@WebFilter(urlPatterns = "/api/*", filterName = "myFilter")
//@ServletComponentScan
结果和三.1.e一样,说明这两种方式都可以。
http://localhost:8080/api/user/filter


http://localhost:8080/test


采用第二种方式进行 过滤用户是否登录成功
修改 application.properties 加入开发接口通配地址
#凡是请求地址层级带有 open 都放行
open:
url: /**/open/**
package com.feng.springboot_filter.filter;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.core.annotation.Order;
import org.springframework.util.AntPathMatcher;
import org.springframework.util.PathMatcher;
import org.springframework.util.StringUtils;
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
//@WebFilter(urlPatterns = "/api/*", filterName = "myFilter")
@Order(1) //指定过滤器的执行顺序,值越大越靠后执行
public class MyFilter implements Filter {
@Value("${open.url}")
private String openUrl;
@Override
public void init(FilterConfig filterConfig) throws ServletException {
System.out.println("初始化过滤器");
Filter.super.init(filterConfig);
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) servletRequest;
String requestURI = request.getRequestURI();
String method = request.getMethod();
System.out.println("拦截器 MyFilter 拦截了请求:" + requestURI + ",方法为:" + method);
// 首先校验是否是开放 api
// 是则:直接放行,否则:再校验token
PathMatcher matcher = new AntPathMatcher();
if (matcher.match(openUrl, requestURI)) {
filterChain.doFilter(servletRequest, servletResponse);
} else {
String token = request.getHeader("token");
if (StringUtils.isEmpty(token)) {
// 若无 token,则转发到:未登录请求
servletRequest.getRequestDispatcher("/api/open/unLogin").forward(servletRequest, servletResponse);
} else {
// 若有 则放行
filterChain.doFilter(servletRequest, servletResponse);
}
}
//filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() {
Filter.super.destroy();
}
}
@GetMapping("/open/home/info")
public Map<String, String> getHome() {
Map<String, String> map = new HashMap<>();
map.put("游客", "欢迎访问首页");
return map;
}
@GetMapping("/open/unLogin")
public String getUnauthorized() {
return "登录失效,请重新登录";
}
package com.feng.springboot_filter.controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import java.util.HashMap;
import java.util.Map;
@RestController
@RequestMapping("/api")
public class FilterController {
@GetMapping("/user/filter")
public String hello() {
return "已经通过了过滤器";
}
@GetMapping("/open/home/info")
public Map<String, String> getHome() {
Map<String, String> map = new HashMap<>();
map.put("游客", "欢迎访问首页");
return map;
}
@GetMapping("/open/unLogin")
public String getUnauthorized() {
return "登录失效,请重新登录";
}
}
首先访问 开放接口
http://localhost:8080/api/user/filter(直接放行)

访问需权鉴接口:http://localhost:8080/api/user/filter
不带 token

带上 token
