如何在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 源码了解


目录
相关文章
|
13天前
|
XML Java 数据格式
Spring从入门到入土(bean的一些子标签及注解的使用)
本文详细介绍了Spring框架中Bean的创建和使用,包括使用XML配置文件中的标签和注解来创建和管理Bean,以及如何通过构造器、Setter方法和属性注入来配置Bean。
53 9
Spring从入门到入土(bean的一些子标签及注解的使用)
|
3天前
|
Java 测试技术 Windows
咦!Spring容器里为什么没有我需要的Bean?
【10月更文挑战第11天】项目经理给小菜分配了一个紧急需求,小菜迅速搭建了一个SpringBoot项目并完成了开发。然而,启动测试时发现接口404,原因是控制器包不在默认扫描路径下。通过配置`@ComponentScan`的`basePackages`字段,解决了问题。总结:`@SpringBootApplication`默认只扫描当前包下的组件,需要扫描其他包时需配置`@ComponentScan`。
|
13天前
|
Java 开发者 Spring
Spring bean的生命周期详解!
本文详细解析Spring Bean的生命周期及其核心概念,并深入源码分析。Spring Bean是Spring框架的核心,由容器管理其生命周期。从实例化到销毁,共经历十个阶段,包括属性赋值、接口回调、初始化及销毁等。通过剖析`BeanFactory`、`ApplicationContext`等关键接口与类,帮助你深入了解Spring Bean的管理机制。希望本文能助你更好地掌握Spring Bean生命周期。
42 1
|
15天前
|
Java Spring
获取spring工厂中bean对象的两种方式
获取spring工厂中bean对象的两种方式
16 1
|
16天前
|
Java 开发者 Spring
Spring bean的生命周期详解!
本文详细介绍了Spring框架中的核心概念——Spring Bean的生命周期,包括实例化、属性赋值、接口回调、初始化、使用及销毁等10个阶段,并深入剖析了相关源码,如`BeanFactory`、`DefaultListableBeanFactory`和`BeanPostProcessor`等关键类与接口。通过理解这些核心组件,读者可以更好地掌握Spring Bean的管理和控制机制。
50 1
|
5天前
|
XML Java 数据格式
Spring IOC容器的深度解析及实战应用
【10月更文挑战第14天】在软件工程中,随着系统规模的扩大,对象间的依赖关系变得越来越复杂,这导致了系统的高耦合度,增加了开发和维护的难度。为解决这一问题,Michael Mattson在1996年提出了IOC(Inversion of Control,控制反转)理论,旨在降低对象间的耦合度,提高系统的灵活性和可维护性。Spring框架正是基于这一理论,通过IOC容器实现了对象间的依赖注入和生命周期管理。
15 0
|
1月前
|
XML Java 数据格式
spring复习03,注解配置管理bean
Spring框架中使用注解配置管理bean的方法,包括常用注解的标识组件、扫描组件、基于注解的自动装配以及使用注解后的注意事项,并提供了一个基于注解自动装配的完整示例。
spring复习03,注解配置管理bean
|
2天前
|
存储 Docker 容器
docker中挂载数据卷到容器
【10月更文挑战第12天】
13 5
|
7天前
|
存储 运维 云计算
探索Docker容器化:从入门到实践
在这个快速发展的云计算时代,Docker容器化技术正在改变应用的开发、部署和管理方式。本文旨在为初学者提供一个关于Docker的全面入门指南,并通过实践案例展示Docker在实际开发中的应用。我们将一起了解Docker的核心概念、基本操作、网络和存储,以及如何构建和部署一个简单的Web应用。无论你是开发者还是运维人员,本文都会帮助你快速掌握Docker的核心技能。
|
3天前
|
存储 测试技术 开发者
docker中将数据卷挂载到容器
【10月更文挑战第11天】
13 3