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


目录
相关文章
|
29天前
|
存储 Java Spring
【Spring】获取Bean对象需要哪些注解
@Conntroller,@Service,@Repository,@Component,@Configuration,关于Bean对象的五个常用注解
|
29天前
|
存储 Java 应用服务中间件
【Spring】IoC和DI,控制反转,Bean对象的获取方式
IoC,DI,控制反转容器,Bean的基本常识,类注解@Controller,获取Bean对象的常用三种方式
|
1月前
|
XML Java 数据格式
Spring容器Bean之XML配置方式
通过对以上内容的掌握,开发人员可以灵活地使用Spring的XML配置方式来管理应用程序的Bean,提高代码的模块化和可维护性。
67 6
|
1月前
|
XML Java 数据格式
🌱 深入Spring的心脏:Bean配置的艺术与实践 🌟
本文深入探讨了Spring框架中Bean配置的奥秘,从基本概念到XML配置文件的使用,再到静态工厂方式实例化Bean的详细步骤,通过实际代码示例帮助读者更好地理解和应用Spring的Bean配置。希望对你的Spring开发之旅有所助益。
118 3
|
1月前
|
安全 Java 开发者
Spring容器中的bean是线程安全的吗?
Spring容器中的bean默认为单例模式,多线程环境下若操作共享成员变量,易引发线程安全问题。Spring未对单例bean做线程安全处理,需开发者自行解决。通常,Spring bean(如Controller、Service、Dao)无状态变化,故多为线程安全。若涉及线程安全问题,可通过编码或设置bean作用域为prototype解决。
43 1
|
1月前
|
监控 NoSQL 时序数据库
《docker高级篇(大厂进阶):7.Docker容器监控之CAdvisor+InfluxDB+Granfana》包括:原生命令、是什么、compose容器编排,一套带走
《docker高级篇(大厂进阶):7.Docker容器监控之CAdvisor+InfluxDB+Granfana》包括:原生命令、是什么、compose容器编排,一套带走
271 77
|
15天前
|
Ubuntu NoSQL Linux
《docker基础篇:3.Docker常用命令》包括帮助启动类命令、镜像命令、有镜像才能创建容器,这是根本前提(下载一个CentOS或者ubuntu镜像演示)、容器命令、小总结
《docker基础篇:3.Docker常用命令》包括帮助启动类命令、镜像命令、有镜像才能创建容器,这是根本前提(下载一个CentOS或者ubuntu镜像演示)、容器命令、小总结
97 6
《docker基础篇:3.Docker常用命令》包括帮助启动类命令、镜像命令、有镜像才能创建容器,这是根本前提(下载一个CentOS或者ubuntu镜像演示)、容器命令、小总结
|
1月前
|
监控 Docker 容器
在Docker容器中运行打包好的应用程序
在Docker容器中运行打包好的应用程序
|
26天前
|
Ubuntu Linux 开发工具
docker 是什么?docker初认识之如何部署docker-优雅草后续将会把产品发布部署至docker容器中-因此会出相关系列文章-优雅草央千澈
Docker 是一个开源的容器化平台,允许开发者将应用程序及其依赖项打包成标准化单元(容器),确保在任何支持 Docker 的操作系统上一致运行。容器共享主机内核,提供轻量级、高效的执行环境。本文介绍如何在 Ubuntu 上安装 Docker,并通过简单步骤验证安装成功。后续文章将探讨使用 Docker 部署开源项目。优雅草央千澈 源、安装 Docker 包、验证安装 - 适用场景:开发、测试、生产环境 通过以上步骤,您可以在 Ubuntu 系统上成功安装并运行 Docker,为后续的应用部署打下基础。
docker 是什么?docker初认识之如何部署docker-优雅草后续将会把产品发布部署至docker容器中-因此会出相关系列文章-优雅草央千澈
|
16天前
|
Kubernetes Linux 虚拟化
入门级容器技术解析:Docker和K8s的区别与关系
本文介绍了容器技术的发展历程及其重要组成部分Docker和Kubernetes。从传统物理机到虚拟机,再到容器化,每一步都旨在更高效地利用服务器资源并简化应用部署。容器技术通过隔离环境、减少依赖冲突和提高可移植性,解决了传统部署方式中的诸多问题。Docker作为容器化平台,专注于创建和管理容器;而Kubernetes则是一个强大的容器编排系统,用于自动化部署、扩展和管理容器化应用。两者相辅相成,共同推动了现代云原生应用的快速发展。
84 11