@Component和@Bean使用注意

@Component和@Bean使用注意,第1张

@Component和@Bean使用注意

  大家都知道@Component和@Bean是spring生成bean对象的注解,@Component只可以加在类上,如果该类在spring的扫描路径之下就可以生成bean对象,@Bean一般与@Configuration结合使用,指定方法名为bean对象的名称,返回对象为bean对象。
正常情况的使用大家肯定都没有问题,下面列举几种需要注意的情况:
项目结构

1、两个相同名称的类在不同包下加@Component
@Component
public class Bean {
    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "Bean{" +
                "name='" + name + ''' +
                '}';
    }
}

两个Bean类代码如上,启动项目会报错

org.springframework.beans.factory.BeanDefinitionStoreException: Failed to parse configuration class [com.example.demo.DemoApplication]; nested exception is org.springframework.context.annotation.ConflictingBeanDefinitionException: Annotation-specified bean name 'bean' for bean class [com.example.demo.b.Bean] conflicts with existing, non-compatible bean definition of same name and class [com.example.demo.a.Bean]

解决方式:
修改其中一个类的名称,或者在@Component中指定不一样的bean的名称。

2、存在同名的@Bean方法名和@Component类
@Configuration
@ComponentScan("com.example.demo")
public class BeanConfig {
    @Bean
    com.example.demo.a.Bean bean(){
        com.example.demo.a.Bean bean = new com.example.demo.a.Bean();
        bean.setName("A");
        return bean;
    }
}
1、以spring的方式启动
//@SpringBootApplication
public class DemoApplication {

    @Autowired
    ApplicationContext ioc;
    public static void main(String[] args) {
//        SpringApplication.run(DemoApplication.class, args);
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(BeanConfig.class);
        Object bean = applicationContext.getBean("bean");
        System.out.println(bean);
    }
}

结果如下:

 DEBUG org.springframework.beans.factory.support.DefaultListableBeanFactory - Creating shared instance of singleton bean 'bean'
Bean{name='A'}

可以看出@Bean会覆盖@Component结果。

2、以springboot方式启动

结果如下:

***************************
APPLICATION FAILED TO START
***************************

Description:

The bean 'bean', defined in class path resource [com/example/demo/config/BeanConfig.class], could not be registered. A bean with that name has already been defined in file [D:learn-masterdemotargetclassescomexampledemobBean.class] and overriding is disabled.

Action:

Consider renaming one of the beans or enabling overriding by setting spring.main.allow-bean-definition-overriding=true

根据提示可以看出需要在配置文件增加spring.main.allow-bean-definition-overriding=true

3、如果存在多个@Bean方法名相同重载
@Configuration
@ComponentScan("com.example.demo")
public class BeanConfig {
    @Bean
    com.example.demo.a.Bean bean(){
        com.example.demo.a.Bean bean = new com.example.demo.a.Bean();
        bean.setName("A");
        return bean;
    }
    @Bean
    com.example.demo.a.Bean bean(com.example.demo.a.Bean bean1, com.example.demo.a.Bean bean2){
        com.example.demo.a.Bean bean = new com.example.demo.a.Bean();
        bean.setName("B");
        return bean;
    }
    @Bean
    com.example.demo.a.Bean bean(com.example.demo.a.Bean bean1){
        com.example.demo.a.Bean bean = new com.example.demo.a.Bean();
        bean.setName("C");
        return bean;
    }
}

  虽然有三个个@Bean,但是肯定只会生成一个bean的Bean,那么Spring在处理@Bean时,也只会生成一个bean的BeanDefinition,比如Spring先解析到第一个@Bean,会生成一个BeanDefinition,此时isFactoryMethodUnique为true,但是解析到第二个@Bean时,会判断出来beanDefinitionMap中已经存在一个bean的BeanDefinition了,那么会把之前的这个BeanDefinition的isFactoryMethodUnique修改为false,并且不会生成新的BeanDefinition了。
  并且后续在根据BeanDefinition创建Bean时,会根据isFactoryMethodUnique来 *** 作,如果为true,那就表示当前BeanDefinition只对应了一个方法,那也就是只能用这个方法来创建Bean了,但是如果isFactoryMethodUnique为false,那就表示当前BeanDefition对应了多个方法,用推断构造方法的逻辑,去选择用哪个方法来创建Bean。
  其属性会将spring的bean自动注入。
  这个 *** 作是不会报错的。

欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/zaji/5697854.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-12-17
下一篇 2022-12-17

发表评论

登录后才能评论

评论列表(0条)

保存