如何在Filter中使用Spring容器中的Bean?

简介: 如何在Filter中使用Spring容器中的Bean?

【1】XML配置形式

也就是在过滤器中使用Spring容器中的bean。

我们使用过滤器一般是这么配置的:

<filter>  
  <filter-name>sysVisitFilter</filter-name>  
  <filter-class>com.hh.core.filter.SysVisitFilter</filter-class> 
</filter>  
<filter-mapping>
  <filter-name>sysVisitFilter</filter-name>
  <url-pattern>/*</url-pattern>
</filter-mapping>

这样的配置无法在过滤器中使用spring bean,因为filter比bean先加载。

也就是spring会先加载filter指定的类到container中,这样filter中注入的spring bean就为null了。


正确配置如下:

<filter>  
  <filter-name>sysVisitFilter</filter-name>  
  <filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class> 
  <init-param>  
   <param-name>targetFilterLifecycle</param-name>  
   <param-value>true</param-value>  
  </init-param>  
</filter>  
<filter-mapping>
  <filter-name>sysVisitFilter</filter-name>
  <url-pattern>/*</url-pattern>
</filter-mapping>

Spring 配置文件中配置该bean:

<bean id="sysVisitFilter" class="com.hh.core.filter.SysVisitFilter"></bean>


DelegatingFilterProxy类遵循filter-name-bean的原则,会根据web.xml中filter-name的值查找到spring配置文件中id与filter-name相同的值,然后把接受到的处理信息传递给相对应的类处理。

另外关于targetFilterLifecycle,查看DelegatingFilterProxy源码部分如下:

 // 设置是否反射调用目标过滤器的init和destroy方法,默认为false-目标bean依赖spring应用上下文管理生命周期
 // 如果设置为true意味着servlet容器将会控制目标Filter的生命周期,使用代理委派相应的调用
public void setTargetFilterLifecycle(boolean targetFilterLifecycle) {
  this.targetFilterLifecycle = targetFilterLifecycle;
}

测试结果如下:



【2】SpringBoot2.0中使用代码注册Filter

SysVisitFilter 示例如下:

public class SysVisitFilter implements Filter {
    private static final Logger log = LoggerFactory.getLogger(SysVisitFilter.class);
    @Autowired
    ISysVisitLogService sysVisitLogService;
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        log.debug("SysVisitFilter is to be inited...");
    }
    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        HttpServletRequest httpRequest = (HttpServletRequest) request;
        String param = httpRequest.getParameter("param");
        String mobileNum = httpRequest.getParameter("mobileNum");
        String remoteIP = IpUtils.getRealRemoteIP(httpRequest);
        try {
            if(ToolUtils.isNotEmpty(param)){
                param = AesUtils.decryptBase64(param, DataInterfaceUtil.aesKey);
                log.debug("the param is :"+param);
                Map<String, Object> mapFromParam = ToolUtils.getMapFromParam(param);
                mobileNum = (String) mapFromParam.get("mobileNum");
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
        StringBuffer url = httpRequest.getRequestURL();
        SysVisitLog sysVisitLog = new SysVisitLog();
        sysVisitLog.setMobileNum(mobileNum);
        sysVisitLog.setUrl(url.toString());
        sysVisitLog.setRemoteIp(remoteIP);
        sysVisitLog.setCreateTime(new Date());
        sysVisitLogService.insertVisitLog(sysVisitLog);
        log.debug("即将要插入的访问日志 :"+sysVisitLog);
        chain.doFilter(request,response);
    }
    @Override
    public void destroy() {
    }
}

MyFilterConfig 编码注册如下:

@Configuration
public class MyFilterConfig {
    @Bean(name="sysVisitFilter")
    public SysVisitFilter sysVisitFilter(){
        return new SysVisitFilter();
    }
    @Bean
    public FilterRegistrationBean sysVisitFilterBean() {
        FilterRegistrationBean registrationBean = new FilterRegistrationBean();
        registrationBean.setFilter(new DelegatingFilterProxy("sysVisitFilter"));
        registrationBean.addInitParameter("targetFilterLifecycle","true");
        registrationBean.addUrlPatterns("/*");
        registrationBean.addInitParameter("exclusions", "*.js,*.gif,*.jpg,*.png,*.css,*.ico");
        registrationBean.setDispatcherTypes(DispatcherType.REQUEST);
        // 这里这里这里
        registrationBean.setEnabled(false);
        return registrationBean;
    }
}

需要额外注意下面这行代码:

registrationBean.setEnabled(false);

如果没有该代码,系统中的过滤器链如下图:


delegatingFilterProxysysVisitFilter同时存在,会发生什么现象?你的请求会在sysVisitFilter被拦截两次!!!


如果添加上该行代码,系统中的过滤器链如下图:

 registrationBean.setEnabled(false);

sysVisitFilter不但能正常使用,而且还能获取到Spring容器中的Bean!!!

更多详情可以查看DelegatingFilterProxy 源码了解


目录
相关文章
|
1月前
|
缓存 安全 Java
《深入理解Spring》过滤器(Filter)——Web请求的第一道防线
Servlet过滤器是Java Web核心组件,可在请求进入容器时进行预处理与响应后处理,适用于日志、认证、安全、跨域等全局性功能,具有比Spring拦截器更早的执行时机和更广的覆盖范围。
|
1月前
|
XML Java 测试技术
《深入理解Spring》:IoC容器核心原理与实战
Spring IoC通过控制反转与依赖注入实现对象间的解耦,由容器统一管理Bean的生命周期与依赖关系。支持XML、注解和Java配置三种方式,结合作用域、条件化配置与循环依赖处理等机制,提升应用的可维护性与可测试性,是现代Java开发的核心基石。
|
1月前
|
XML Java 应用服务中间件
【SpringBoot(一)】Spring的认知、容器功能讲解与自动装配原理的入门,带你熟悉Springboot中基本的注解使用
SpringBoot专栏开篇第一章,讲述认识SpringBoot、Bean容器功能的讲解、自动装配原理的入门,还有其他常用的Springboot注解!如果想要了解SpringBoot,那么就进来看看吧!
358 2
|
9月前
|
XML Java 测试技术
Spring IOC—基于注解配置和管理Bean 万字详解(通俗易懂)
Spring 第三节 IOC——基于注解配置和管理Bean 万字详解!
646 26
|
XML Java 数据格式
京东一面:spring ioc容器本质是什么? ioc容器启动的步骤有哪些?
京东一面:spring ioc容器本质是什么? ioc容器启动的步骤有哪些?
|
6月前
|
XML Java 数据格式
Spring IoC容器的设计与实现
Spring 是一个功能强大且模块化的 Java 开发框架,其核心架构围绕 IoC 容器、AOP、数据访问与集成、Web 层支持等展开。其中,`BeanFactory` 和 `ApplicationContext` 是 Spring 容器的核心组件,分别定位为基础容器和高级容器,前者提供轻量级的 Bean 管理,后者扩展了事件发布、国际化等功能。
|
9月前
|
XML Java 数据格式
Spring容器的本质
本文主要讨论Spring容器最核心的机制,用最少的代码讲清楚Spring容器的本质。
|
11月前
|
存储 Java Spring
【Spring】获取Bean对象需要哪些注解
@Conntroller,@Service,@Repository,@Component,@Configuration,关于Bean对象的五个常用注解
296 12
|
11月前
|
存储 Java 应用服务中间件
【Spring】IoC和DI,控制反转,Bean对象的获取方式
IoC,DI,控制反转容器,Bean的基本常识,类注解@Controller,获取Bean对象的常用三种方式
400 12
|
11月前
|
XML Java 数据格式
Spring容器Bean之XML配置方式
通过对以上内容的掌握,开发人员可以灵活地使用Spring的XML配置方式来管理应用程序的Bean,提高代码的模块化和可维护性。
357 6

热门文章

最新文章