ImportBeanDefinitionRegistrar动态注册bean

简介

  • ImportBeanDefinitionRegistrar类只能通过其他类@Import的方式来加载,通常是启动类或配置类。
  • 使用@Import,如果括号中的类是ImportBeanDefinitionRegistrar的实现类,则会调用接口方法,将其中要注册的类注册成bean。
  • 实现该接口的类拥有注册bean的能力。

手动把一个类注册成bean

  • 首先写一个类,最终要把它注册为bean。

      public class HelloService {
    
      }
    
  • 自定义ImportBeanDefinitionRegistrar实现类手动注册bean。

      public class HelloImportBeanDefinitionRegistrar 
              implements ImportBeanDefinitionRegistrar {
    
          /**
           * @Author haien
           * @Description AnnotationMetadata:当前类的注解信息;
           * BeanDefinitionRegistry:注册类,其registerBeanDefinition()可以注册bean
           * @Date 2019/6/12
           * @Param [importingClassMetadata, registry]
           * @return void
           **/
          @Override
          public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata,
                                              BeanDefinitionRegistry registry) {
      
              //扫描注解
              Map<String, Object> annotationAttributes = importingClassMetadata
                  .getAnnotationAttributes(ComponentScan.class.getName());
              String[] basePackages = (String[]) annotationAttributes.get("basePackages");
      
              //扫描类
              ClassPathBeanDefinitionScanner scanner =
                      new ClassPathBeanDefinitionScanner(registry, false);
              TypeFilter helloServiceFilter = new AssignableTypeFilter(HelloService.class);
              
              scanner.addIncludeFilter(helloServiceFilter);
              scanner.scan(basePackages);
          }
      
      }
    
  • 最后定义一个配置类发现一下上面的ImportBeanDefinitionRegistrar实现类。

      @Configuration
      @ComponentScan("com.haien.import2.domain")
      @Import(HelloImportBeanDefinitionRegistrar.class)
      public class HelloConfiguration {
      
      }
    
  • 测试:

      @RunWith(SpringRunner.class)
      @SpringBootTest
      @ContextConfiguration(classes = {HelloConfiguration.class}) //表示只需要这一个文件
      public class DemoApplicationTest2 {
      
          @Resource
          HelloService helloService;
      
          /**
           * @Author haien
           * @Description 扫描到helloService
           * @Date 2019/6/11
           * @Param []
           * @return void
           **/
          @Test
          public void contextLoads(){
              System.out.println(helloService.getClass());
          }
      }
    

自己写一个注解实现@Component的功能

  • 目标:自己写一个注解@Mapper,配合其他类,实现被注解类可以被注册成bean的功能。

  • @Mapper:

      @Documented
      @Inherited
      @Retention(RetentionPolicy.RUNTIME)
      @Target({ElementType.TYPE,ElementType.FIELD,ElementType.METHOD,ElementType.PARAMETER})
      public @interface Mapper {
      }
    
  • 注释于类上:

      @Mapper
      public class CountryMapper {
      }
    
  • 实现ImportBeanDefinitionRegistrar接口,重写registerBeanDefinitions方法,手动注册bean;同时实现一些Aware接口,以便获取Spring的一些数据。

      public class MapperAutoConfiguredMyBatisRegistrar implements
              ImportBeanDefinitionRegistrar,ResourceLoaderAware,BeanFactoryAware {
          
          private ResourceLoader resourceLoader;
          private BeanFactory beanFactory;
      
          /**
           * @Author haien
           * @Description 注册bean,但我们并不知道需要注册哪些bean,所以需要借助
           * ClassPathBeanDefinitionScanner扫描器,扫描我们需要注册的bean
           * @Date 2019/6/11
           * @Param [annotationMetadata, beanDefinitionRegistry]
           * @return void
           **/
          @Override
          public void registerBeanDefinitions(AnnotationMetadata annotationMetadata,
                                              BeanDefinitionRegistry registry) {
              
              MyClasssPathBeanDefinitionScanner scanner=
                      new MyClasssPathBeanDefinitionScanner(registry,false);
              scanner.setResourceLoader(resourceLoader);
              scanner.registerFilters();
              scanner.doScan("com.haien.import1.domain");
          }
      
          @Override
          public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
              this.beanFactory=beanFactory;
          }
      
          @Override
          public void setResourceLoader(ResourceLoader resourceLoader) {
              this.resourceLoader=resourceLoader;
          }
      
      }
    
  • 以上我们还借助了扫描器ClassPathBeanDefinitionScanner,通过它来获取我们需要注册的bean。

      public class MyClasssPathBeanDefinitionScanner extends ClassPathBeanDefinitionScanner {
    
          public MyClasssPathBeanDefinitionScanner(BeanDefinitionRegistry registry,
                                                   boolean useDefaultFilters) {
              super(registry, useDefaultFilters);
          }
      
          /**
           * @Author haien
           * @Description 注册条件过滤器,将@Mapper注解加入允许扫描的过滤器中,
           * 如果加入排除扫描的过滤器集合excludeFilter中,则不会扫描
           * @Date 2019/6/11
           * @Param []
           * @return void
           **/
          protected void registerFilters(){
              addIncludeFilter(new AnnotationTypeFilter(Mapper.class));
          }
      
          @Override
          protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
              return super.doScan(basePackages);
          }
      }
    
  • 测试:

      @RunWith(SpringRunner.class)
      @SpringBootTest(classes = MapperAutoConfig.class)
      //只将MapperAutoConfig类纳入测试环境的Spring容器中,
      //或@ContextConfiguration(classes = {MapperAutoConfig.class})
      public class DemoApplicationTest {
      
          @Resource
          CountryMapper countryMapper;
      
          /**
           * @Author haien
           * @Description 扫描不到CountryMapper
           * @Date 2019/6/11
           * @Param []
           * @return void
           **/
          @Test
          public void contextLoads(){
              System.out.println(countryMapper.getClass());
          }
      }
    
  • 手动把一个类注册成bean

  • 自己写一个注解实现@Component的功能

  • 代码示例:ideaProjects/dynamic-switch-datasource/import2、import1

### 回答1: Spring Boot提供了多种方式来注册bean,其中一种是手动注册bean。具体步骤如下: 1. 创建一个Java类,用于定义要注册bean。 2. 在该类上添加注解@Component或其他相关注解,以便Spring Boot能够扫描到该类。 3. 在该类中定义一个方法,用于创建要注册bean。 4. 在该方法上添加注解@Bean,以便告诉Spring Boot该方法返回的对象应该被注册为一个bean。 5. 在需要使用该bean的地方,使用@Autowired或其他相关注解将该bean注入到目标类中。 需要注意的是,手动注册bean需要在Spring Boot启动时进行,可以通过在启动类中添加注解@Import来引入要注册bean类。 ### 回答2: 在Spring Boot中,可以通过手动注册bean的方式来将自定义的类纳入Spring容器管理。 首先,我们需要在一个配置类中添加`@Configuration`注解,将其声明为配置类。然后,使用`@Bean`注解来标记一个方法,该方法的返回值将作为一个Spring bean进行注册。 下面是一个示例: ```java @Configuration public class MyConfiguration { @Bean public MyBean myBean() { return new MyBean(); } // 其他自定义配置及bean的注册方法 } ``` 在上述示例中,`MyConfiguration`类被声明为一个配置类,并且通过`@Bean`注解将`myBean()`方法返回的实例注册为一个Spring bean。 需要注意的是,被手动注册bean可以被其他组件自动注入或通过`@Autowired`注解来获取。 除了使用上述方式手动注册bean,Spring Boot还提供了其他的方式来注册bean,例如使用`@ComponentScan`、`@Import`、`@ImportResource`等注解,或者实现`ImportBeanDefinitionRegistrar`、`BeanFactoryPostProcessor`等自定义接口。 总的来说,Spring Boot中手动注册bean的方式很灵活,可以根据实际需求选择最适合的方式进行注册。 ### 回答3: 在Spring Boot中手动注册bean可以通过编写配置类(Configuration Class)来实现。 首先,创建一个配置类,可以使用`@Configuration`注解进行标记。在配置类中,可以使用`@Bean`注解来定义需要注册bean。例如: ```java @Configuration public class MyConfiguration { @Bean public MyBean myBean() { return new MyBean(); } // 可以同时定义多个bean // ... } ``` 上述例子中,`myBean()`方法使用`@Bean`注解定义了一个`MyBean`类的实例,然后在应用中就可以通过自动装配或者手动获取来使用该bean了。 接下来,需要在Spring Boot应用的主类上使用`@Import`注解来引入该配置类。例如: ```java @SpringBootApplication @Import(MyConfiguration.class) public class MyApplication { public static void main(String[] args) { SpringApplication.run(MyApplication.class, args); } } ``` 上述例子中,`@Import`注解将`MyConfiguration`配置类引入到应用中,这样在启动应用时,定义的bean就会被注册到Spring容器中。 需要注意的是,手动注册bean一般适用于一些特殊情况,不推荐频繁使用该方法,因为Spring Boot已经提供了自动配置的功能,可以根据约定和配置进行自动注册bean,可以更好地减少开发工作量和提高开发效率。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值