本文的大纲如下
这周遇到了这样一个需求,从第三方的数据库中获取值,只是一个简单的分页查询,处理这种问题,我一般都是在配置文件中配置数据库的地址等相关信息,然后在Spring Configuration 注册数据量连接池的bean,然后再将数据库连接池给JdbcTemplate, 但是这种的缺陷是,假设填错了数据库地址和密码,或者换了数据库的地址和密码,在配置文件里面重启之后,都需要重启应用。
我想能不能动态的向Spring IOC容器中注册和加载bean呢,项目在界面上填写数据库的地址、用户名、密码,存储之后,将JdbcTemplate和另一个数据库连接池加载到IOC容器中。答案是可以的,我经过一番搜索写出了如下代码:
@Componentpublic class BeanDynamicRegister { private final ConfigurableApplicationContext configurableApplicationContext; public BeanDynamicRegister(ConfigurableApplicationContext configurableApplicationContext) { this.configurableApplicationContext = configurableApplicationContext; } /** * 此方法提供出去,供其他bean动态的向IOC容器中注册bean。 * 代表使用构造器给bean赋值 * * @param beanName bean名 * @param clazz bean类 * @param args 用于向bean的构造函数中添加值 如果loadType是set,则要求传递map.map的key为属性名,value为属性值 * @param 返回一个泛型 * @param loadType * @return */ public T registerBeanByLoadType(String beanName, Class clazz, LoadType loadType, Object... args) { BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(clazz); if (args.length > 0) { // 将参数加入到构造函数中 switch (loadType) { case CONSTRUCTOR: for (Object arg : args) { beanDefinitionBuilder.addConstructorArgValue(arg); } break; case SETTER: Map propertyMap = (Map) args[0]; for (Map.Entry stringObjectEntry : propertyMap.entrySet()) { beanDefinitionBuilder.addPropertyValue(stringObjectEntry.getKey(), stringObjectEntry.getValue()); } break; default: break; } } BeanDefinition beanDefinition = beanDefinitionBuilder.getRawBeanDefinition(); BeanDefinitionRegistry beanDefinitionRegistry = (BeanDefinitionRegistry) configurableApplicationContext.getBeanFactory(); beanDefinitionRegistry.registerBeanDefinition(beanName, beanDefinition); return configurableApplicationContext.getBean(beanName, clazz); } public T getBeanByName(String beanName,Class requiredType){ return configurableApplicationContext.getBean(beanName,requiredType); } /** * 如果用户换了地址和密码,向IOC容器中移除bean。 重新注册 * * @param beanName */ public void removeBean(String beanName) { BeanDefinitionRegistry beanDefinitionRegistry = (BeanDefinitionRegistry) configurableApplicationContext.getBeanFactory(); beanDefinitionRegistry.removeBeanDefinition(beanName); }}@SpringBootTestclass SsmApplicationTests { @Autowired private LoadBeanService loadBeanService; private NamedParameterJdbcTemplate jdbcTemplate; @Autowired private BeanDynamicRegister beanDynamicRegister; @Test public void test() { loadBeanService.loadDataSourceTest("root", "root"); jdbcTemplate = beanDynamicRegister.getBeanByName("jdbcTemplateOne", NamedParameterJdbcTemplate.class); System.out.println("--------" + jdbcTemplate); }}
结果:
我们就到这里了吗? 我们观察一下上面将一个bean加载到Spring IOC容器里经过了几步:
联系我们前面的文章《Spring Bean 的生命周期》,我们将Spring 的生命周期理解为“Spring 给我们提供的一些扩展接口,如果bean实现了这些这些接口,应用在启动的过程中会回调这些接口的方法。” , 这个理解并不完善,缺少了解析BeanDefinition这个阶段。
那BeanDefinition是什么? BeanDefinition是一个接口,我们进Spring 官网( https://docs.spring.io/spring... )大致看一下:
A bean definition can contain a lot of configuration information, including constructor arguments, property values, and container-specific information, such as the initialization method, a static factory method name, and so on. A child bean definition inherits configuration data from a parent definition. The child definition can override some values or add others as needed. Using parent and child bean definitions can save a lot of typing. Effectively, this is a form of templating.
这段说的可能有点抽象, 你点BeanDefinition进去,你就会发现有很多熟悉的面孔:
Bean的作用域: 单例,还是多例。
lazyInit是否是懒加载。
这些都是描述Spring Bean的信息,我们可以类比到Java中的类,每个类都会有class属性,我们在配置类或者xml中的配置Bean的元信息,也被映射到这里。供IOC容器将Bean加入时使用。所以我们可以为对Spring Bean的生命周期的理解打一个补丁:
我们可以打断点来验证一下:
我们这里再来总结一下一个Bean注入Spring IOC容器的几种形式:
BeanDefinitionRegistryPostProcessor也是一个接口,留给我们实现的方法如下:
void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException;
有种越学越不会的感觉。
留言与评论(共有 0 条评论) “” |