大家都知道@Component和@Bean是spring生成bean对象的注解,@Component只可以加在类上,如果该类在spring的扫描路径之下就可以生成bean对象,@Bean一般与@Configuration结合使用,指定方法名为bean对象的名称,返回对象为bean对象。
正常情况的使用大家肯定都没有问题,下面列举几种需要注意的情况:
项目结构
@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的名称。
@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自动注入。
这个 *** 作是不会报错的。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)