前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >bean放入Spring容器

bean放入Spring容器

原创
作者头像
ruochen
修改2021-11-23 13:42:12
5360
修改2021-11-23 13:42:12
举报

@Configuration + @Bean

这种方式其实,在上一篇文章已经介绍过了,也是我们最常用的一种方式,@Configuration用来声明一个配置类,然后使用 @Bean

注解,用于声明一个bean,将其加入到Spring容器中。

具体代码如下:

代码语言:txt
复制
@Configuration
代码语言:txt
复制
public class MyConfiguration {
代码语言:txt
复制
    @Bean
代码语言:txt
复制
    public Person person() {
代码语言:txt
复制
        Person person = new Person();
代码语言:txt
复制
        person.setName("spring");
代码语言:txt
复制
        return person;
代码语言:txt
复制
    }
代码语言:txt
复制
}

@Componet + @ComponentScan

这种方式也是我们用的比较多的方式,@Componet中文译为组件,放在类名上面,然后@ComponentScan放置在我们的配置类上,然后可以指定一个路径,进行扫描带有@Componet注解的bean,然后加至容器中。

具体代码如下:

代码语言:txt
复制
@Component
代码语言:txt
复制
public class Person {
代码语言:txt
复制
    private String name;
代码语言:txt
复制
    public String getName() {
代码语言:txt
复制
        return name;
代码语言:txt
复制
    }
代码语言:txt
复制
    public void setName(String name) {
代码语言:txt
复制
        this.name = name;
代码语言:txt
复制
    }
代码语言:txt
复制
    @Override
代码语言:txt
复制
    public String toString() {
代码语言:txt
复制
        return "Person{" +
代码语言:txt
复制
                "name='" + name + '\'' +
代码语言:txt
复制
                '}';
代码语言:txt
复制
    }
代码语言:txt
复制
}
代码语言:txt
复制
@ComponentScan(basePackages = "it.chusen.spring.*")
代码语言:txt
复制
public class Demo1 {
代码语言:txt
复制
    public static void main(String[] args) {
代码语言:txt
复制
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(Demo1.class);
代码语言:txt
复制
        Person bean = applicationContext.getBean(Person.class);
代码语言:txt
复制
        System.out.println(bean);
代码语言:txt
复制
    }
代码语言:txt
复制
}

结果输出:

代码语言:txt
复制
Person{name='null'}

表示成功将Person放置在了IOC容器中。

@Import注解导入

前两种方式,大家用的可能比较多,也是平时开发中必须要知道的,@Import注解用的可能不是特别多了,但是也是非常重要的,在进行Spring扩展时经常会用到,它经常搭配自定义注解进行使用,然后往容器中导入一个配置文件。关于@Import注解,我会多介绍一点,它有三种使用方式,我会一一介绍。这是@Import注解的源码,表示只能放置在类上。

代码语言:txt
复制
@Target(ElementType.TYPE)
代码语言:txt
复制
@Retention(RetentionPolicy.RUNTIME)
代码语言:txt
复制
@Documented
代码语言:txt
复制
public @interface Import {
代码语言:txt
复制
    /**
代码语言:txt
复制
   * 用于导入一个class文件
     * {@link Configuration @Configuration}, {@link ImportSelector},
     * {@link ImportBeanDefinitionRegistrar}, or regular component classes to import.
     */
    Class<?>[] value();
代码语言:txt
复制
}

@Import直接导入类

代码示例如下:

代码语言:txt
复制
public class Person {
代码语言:txt
复制
    private String name;
代码语言:txt
复制
    public String getName() {
代码语言:txt
复制
        return name;
代码语言:txt
复制
    }
代码语言:txt
复制
    public void setName(String name) {
代码语言:txt
复制
        this.name = name;
代码语言:txt
复制
    }
代码语言:txt
复制
    @Override
代码语言:txt
复制
    public String toString() {
代码语言:txt
复制
        return "Person{" +
代码语言:txt
复制
                "name='" + name + '\'' +
代码语言:txt
复制
                '}';
代码语言:txt
复制
    }
代码语言:txt
复制
}
代码语言:txt
复制
/**
代码语言:txt
复制
* 直接使用@Import导入person类,然后尝试从applicationContext中取,成功拿到
**/
@Import(Person.class)
public class Demo1 {
代码语言:txt
复制
    public static void main(String[] args) {
代码语言:txt
复制
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(Demo1.class);
代码语言:txt
复制
        Person bean = applicationContext.getBean(Person.class);
代码语言:txt
复制
        System.out.println(bean);
代码语言:txt
复制
    }
代码语言:txt
复制
}

上述代码直接使用@Import导入了一个类,然后自动的就被放置在IOC容器中了。 注意

我们的Person类上 就不需要任何的注解了,直接导入即可。

@Import + ImportSelector

其实在@Import注解的源码中,说的已经很清楚了,感兴趣的可以看下,我们实现一个ImportSelector的接口,然后实现其中的方法,进行导入。

代码如下:

代码语言:txt
复制
@Import(MyImportSelector.class)
代码语言:txt
复制
public class Demo1 {
代码语言:txt
复制
    public static void main(String[] args) {
代码语言:txt
复制
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(Demo1.class);
代码语言:txt
复制
        Person bean = applicationContext.getBean(Person.class);
代码语言:txt
复制
        System.out.println(bean);
代码语言:txt
复制
    }
代码语言:txt
复制
}
代码语言:txt
复制
class MyImportSelector implements ImportSelector {
代码语言:txt
复制
    @Override
代码语言:txt
复制
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {
代码语言:txt
复制
        return new String[]{"it.chusen.spring.model.Person"};
代码语言:txt
复制
    }
代码语言:txt
复制
}

我自定义了一个 MyImportSelector 实现了 selectImports

方法,然后将我们要导入的类的全限定名写在里面即可,实现起来也是非常简单。

@Import + ImportBeanDefinitionRegistrar

这种方式也需要我们实现 ImportBeanDefinitionRegistrar 中的方法,具体代码如下:

代码语言:txt
复制
@Import(MyImportBeanDefinitionRegistrar.class)
代码语言:txt
复制
public class Demo1 {
代码语言:txt
复制
    public static void main(String[] args) {
代码语言:txt
复制
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(Demo1.class);
代码语言:txt
复制
        Person bean = applicationContext.getBean(Person.class);
代码语言:txt
复制
        System.out.println(bean);
代码语言:txt
复制
    }
代码语言:txt
复制
}
代码语言:txt
复制
class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {
代码语言:txt
复制
    @Override
代码语言:txt
复制
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
代码语言:txt
复制
        // 构建一个beanDefinition, 关于beanDefinition我后续会介绍,可以简单理解为bean的定义.
代码语言:txt
复制
        AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder.rootBeanDefinition(Person.class).getBeanDefinition();
代码语言:txt
复制
        // 将beanDefinition注册到Ioc容器中.
代码语言:txt
复制
        registry.registerBeanDefinition("person", beanDefinition);
代码语言:txt
复制
    }
代码语言:txt
复制
}

上述实现其实和Import的第二种方式差不多,都需要去实现接口,然后进行导入。接触到了一个新的概念,BeanDefinition,可以简单理解为bean的定义(bean的元数据),也是需要放在IOC容器中进行管理的,先有bean的元数据,applicationContext再根据bean的元数据去创建Bean。

@Import + DeferredImportSelector

这种方式也需要我们进行实现接口,其实它和@Import的第二种方式差不多,DeferredImportSelector 它是 ImportSelector

的子接口,所以实现的方法和第二种无异。只是Spring的处理方式不同,它和Spring Boot中的自动导入配置文件 延迟导入有关, 非常重要

。使用方式如下:

代码语言:txt
复制
@Import(MyDeferredImportSelector.class)
代码语言:txt
复制
public class Demo1 {
代码语言:txt
复制
    public static void main(String[] args) {
代码语言:txt
复制
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(Demo1.class);
代码语言:txt
复制
        Person bean = applicationContext.getBean(Person.class);
代码语言:txt
复制
        System.out.println(bean);
代码语言:txt
复制
    }
代码语言:txt
复制
}
代码语言:txt
复制
class MyDeferredImportSelector implements DeferredImportSelector {
代码语言:txt
复制
    @Override
代码语言:txt
复制
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {
代码语言:txt
复制
        // 也是直接将Person的全限定名放进去
代码语言:txt
复制
        return new String[]{Person.class.getName()};
代码语言:txt
复制
    }
代码语言:txt
复制
}

关于@Import注解的使用方式,大概就以上三种,当然它还可以搭配@Configuration注解使用,用于导入一个配置类。

使用FactoryBean接口

FactoryBean接口和BeanFactory千万不要弄混了,从名字其实可以大概的区分开,FactoryBean,

后缀为bean,那么它其实就是一个bean, BeanFactory,顾名思义 bean工厂,它是IOC容器的顶级接口,这俩接口其实都很重要。

代码示例:

代码语言:txt
复制
@Configuration
代码语言:txt
复制
public class Demo1 {
代码语言:txt
复制
    @Bean
代码语言:txt
复制
    public PersonFactoryBean personFactoryBean() {
代码语言:txt
复制
        return new PersonFactoryBean();
代码语言:txt
复制
    }
代码语言:txt
复制
    public static void main(String[] args) {
代码语言:txt
复制
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(Demo1.class);
代码语言:txt
复制
        Person bean = applicationContext.getBean(Person.class);
代码语言:txt
复制
        System.out.println(bean);
代码语言:txt
复制
    }
代码语言:txt
复制
}
代码语言:txt
复制
class PersonFactoryBean implements FactoryBean<Person> {
代码语言:txt
复制
    /**
代码语言:txt
复制
     *  直接new出来Person进行返回.
     */
    @Override
    public Person getObject() throws Exception {
        return new Person();
    }
    /**
     *  指定返回bean的类型.
     */
    @Override
    public Class<?> getObjectType() {
        return Person.class;
    }
}

上述代码,我使用@Configuration + @Bean的方式将 PersonFactoryBean 加入到容器中,注意,我没有向容器中注入

Person, 而是直接注入的 PersonFactoryBean 然后从容器中拿Person这个类型的bean,成功运行。

使用 BeanDefinitionRegistryPostProcessor

其实这种方式也是利用到了 BeanDefinitionRegistry,在Spring容器启动的时候会执行

BeanDefinitionRegistryPostProcessor 的 postProcessBeanDefinitionRegistry

方法,大概意思就是等beanDefinition加载完毕之后,对beanDefinition进行后置处理,可以在此进行调整IOC容器中的beanDefinition,从而干扰到后面进行初始化bean。

具体代码如下:

代码语言:txt
复制
public class Demo1 {
代码语言:txt
复制
    public static void main(String[] args) {
代码语言:txt
复制
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext();
代码语言:txt
复制
        MyBeanDefinitionRegistryPostProcessor beanDefinitionRegistryPostProcessor = new MyBeanDefinitionRegistryPostProcessor();
代码语言:txt
复制
        applicationContext.addBeanFactoryPostProcessor(beanDefinitionRegistryPostProcessor);
代码语言:txt
复制
        applicationContext.refresh();
代码语言:txt
复制
        Person bean = applicationContext.getBean(Person.class);
代码语言:txt
复制
        System.out.println(bean);
代码语言:txt
复制
    }
代码语言:txt
复制
}
代码语言:txt
复制
class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor {
代码语言:txt
复制
    @Override
代码语言:txt
复制
    public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
代码语言:txt
复制
        AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder.rootBeanDefinition(Person.class).getBeanDefinition();
代码语言:txt
复制
        registry.registerBeanDefinition("person", beanDefinition);
代码语言:txt
复制
    }
代码语言:txt
复制
    @Override
代码语言:txt
复制
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
代码语言:txt
复制
    }
代码语言:txt
复制
}

上述代码中,我们手动向beanDefinitionRegistry中注册了person的BeanDefinition。最终成功将person加入到applicationContext中,上述的几种方式的具体原理,我后面会进行介绍。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
作者已关闭评论
0 条评论
热度
最新
推荐阅读
目录
  • @Configuration + @Bean
  • @Componet + @ComponentScan
  • @Import注解导入
    • @Import直接导入类
      • @Import + ImportSelector
        • @Import + ImportBeanDefinitionRegistrar
          • @Import + DeferredImportSelector
          • 使用FactoryBean接口
          • 使用 BeanDefinitionRegistryPostProcessor
          相关产品与服务
          容器服务
          腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
          领券
          问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档