SpringBoot中动态注册Bean的方式
创始人
2025-01-11 16:04:52
0

测试环境,本文源码

  • Java:8
  • SpringBoot:2.5.14
  • 示例场景:动态注册ProxyServlet,间接实现类似于Nginx的反向代理功能

先理解如何实现动态注册 Bean

由于在 SpringBoot 中,先进行 Bean 的定义,再根据定义进行 Bean 的实例化,所以实现动态 Bean ,我们只需要动态注册 Bean 定义即可。

这就用到了 BeanDefinitionRegistryPostProcessorpostProcessBeanDefinitionRegistry 这个方法。

源码注释

Modify the application context’s internal bean definition registry after its standard initialization. All regular bean definitions will have been loaded, but no beans will have been instantiated yet. This allows for adding further bean definitions before the next post-processing phase kicks in.

所有常规的Bean都已经定义但尚未实例化时,这时候,你可以再新增 Bean 定义

一、通用方式

先说一个小插曲,建议搭配源码食用。假如三个类,他们分别实现且只实现了以下三个接口

  • BeanDefinitionRegistryPostProcessor
  • ApplicationContextAware
  • EnvironmentAware

这时候,这三个 Bean 的默认加载顺序如下。

并且,这三个 Bean 的方法执行顺序也是跟加载顺序相同。

但是,如果 Bean 实现了一个比 ApplicationContextAware 或者 EnvironmentAware 更先加载的 Bean ,那么就会出现先执行 setApplicationContext 或者 setEnvironment 的方法,为啥嘞?

是因为在 Bean 初始化前,先判定有没有实现 Aware 接口,如果实现过了,那么就直接优先调用 Aware 中的方法。如图。

既然明白了这个流程,那我们就可以实现功能了。

该通用方式是适用于 SpringBoot 框架中通用动态注册 Bean 的做法。主要是通过 BeanDefinitionRegistryPostProcessor 动态注册 Bean 定义。

import org.mitre.dsmiley.httpproxy.ProxyServlet; import org.springframework.beans.BeansException; import org.springframework.beans.factory.config.ConfigurableListableBeanFactory; import org.springframework.beans.factory.support.BeanDefinitionBuilder; import org.springframework.beans.factory.support.BeanDefinitionRegistry; import org.springframework.beans.factory.support.BeanDefinitionRegistryPostProcessor; import org.springframework.boot.web.servlet.ServletRegistrationBean; import org.springframework.context.EnvironmentAware; import org.springframework.context.annotation.Configuration; import org.springframework.core.env.Environment;  import java.util.HashMap; import java.util.Map;  /**  * 动态配置bean  *  * @author chenchuancheng  * @since 2024/07/11 22:50  */ @Configuration public class DynamicBeanConfig implements BeanDefinitionRegistryPostProcessor, EnvironmentAware {      private Environment environment;      @Override     public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {         BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition(ServletRegistrationBean.class);         builder.addConstructorArgValue(new ProxyServlet());         builder.addConstructorArgValue(environment.getProperty("proxy.servletUrl"));         Map initParams = new HashMap<>();         initParams.put("targetUri", environment.getProperty("proxy.targetUrl"));         initParams.put("log", "true");         builder.addPropertyValue("initParameters", initParams);         builder.addPropertyValue("name", environment.getProperty("proxy.name"));         registry.registerBeanDefinition("proxyServlet", builder.getBeanDefinition());     }      @Override     public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {     }      @Override     public void setEnvironment(Environment environment) {         this.environment = environment;     } } 

除了实现 Aware 接口,也可以通过构造函数注入,也可以保证调用方法时有值了。

二、特定方式

这个是使用 ServletContext 动态注册 Servlet 的方式。

import org.mitre.dsmiley.httpproxy.ProxyServlet; import org.springframework.boot.web.servlet.ServletContextInitializer; import org.springframework.context.annotation.Configuration; import org.springframework.core.env.Environment;  import javax.servlet.ServletContext; import javax.servlet.ServletException; import javax.servlet.ServletRegistration; import java.util.HashMap; import java.util.Map;  @Configuration public class ProxyServletInitializer implements ServletContextInitializer {      private final Environment environment;      public ProxyServletInitializer(Environment environment) {         this.environment = environment;     }      @Override     public void onStartup(ServletContext servletContext) throws ServletException {         registerProxyServlet(servletContext,environment.getProperty("proxy.name"),environment.getProperty("proxy.targetUrl"));     }      private void registerProxyServlet(ServletContext servletContext, String name, String targetUri) {         ProxyServlet proxyServlet = new ProxyServlet();         ServletRegistration.Dynamic registration = servletContext.addServlet(name + "ProxyServlet", proxyServlet);         registration.setLoadOnStartup(1);         registration.addMapping("/" + name + "/*");                  Map initParameters = new HashMap<>();         initParameters.put("targetUri", targetUri);         initParameters.put("log", "true");         registration.setInitParameters(initParameters);     } } 

相关内容

热门资讯

现有说明如下!wpk俱乐部是真... 现有说明如下!wpk俱乐部是真的吗,we-poker辅助(透视)手段教程(都是存在有挂)1、玩家可以...
无独有偶!约局吧德州有挂吗,w... 无独有偶!约局吧德州有挂吗,wepoker私人局外卦(透视)攻略教程(一直是真的挂)1、下载好约局吧...
长期以来!德普软件,拱趴大菠萝... 长期以来!德普软件,拱趴大菠萝挂哪里(透视)资料教程(一贯真的是有挂)1、完成拱趴大菠萝挂哪里辅助器...
无独有偶!aapoker辅助软... 无独有偶!aapoker辅助软件合法吗,aapoker万能辅助器(透视)模块教程(其实真的是有挂)一...
随着!智星德州可以透视吗,we... 随着!智星德州可以透视吗,wepoker免费脚本弱密码(透视)指南教程(好像真的有挂)1、每一步都需...
更值得关注的是!epoker免... 更值得关注的是!epoker免费透视脚本,悦扑克脚本(透视)手筋教程(切实是有挂)1、进入游戏-大厅...
截至目前!wepokerplu... 截至目前!wepokerplus透视脚本免费,hhpoker辅助挂(透视)讲义教程(好像真的有挂)1...
现有说明如下!wepoker脚... 现有说明如下!wepoker脚本,wepoker辅助软件视频(透视)总结教程(果然存在有挂)1、we...
刚刚!hhpoker可以控制牌... 刚刚!hhpoker可以控制牌吗,wepoker轻量版透视方法(透视)教材教程(其实是真的挂)暗藏猫...
无独有偶!hhpoker是真的... 无独有偶!hhpoker是真的还是假的,wpk德州局怎么透视(透视)阶段教程(好像存在有挂)1.hh...