Spring創(chuàng)建Bean的6種方式詳解
前言
本文講解了在Spring 應(yīng)用中創(chuàng)建Bean的多種方式,包括自動(dòng)創(chuàng)建,以及手動(dòng)創(chuàng)建注入方式,實(shí)際開(kāi)發(fā)中可以根據(jù)業(yè)務(wù)場(chǎng)景選擇合適的方案。
方式1:
使用Spring XML方式配置,該方式用于在純Spring 應(yīng)用中,適用于簡(jiǎn)單的小應(yīng)用,當(dāng)應(yīng)用變得復(fù)雜,將會(huì)導(dǎo)致XMl配置文件膨脹 ,不利于對(duì)象管理。
<bean id="xxxx" class="xxxx.xxxx"/>
方式2:
使用@Component,@Service,@Controler,@Repository注解
這幾個(gè)注解都是同樣的功能,被注解的類將會(huì)被Spring 容器創(chuàng)建單例對(duì)象。
- @Component : 側(cè)重于通用的Bean類
- @Service:標(biāo)識(shí)該類用于業(yè)務(wù)邏輯
- @Controler:標(biāo)識(shí)該類為Spring MVC的控制器類
- @Repository: 標(biāo)識(shí)該類是一個(gè)實(shí)體類,只有屬性和Setter,Getter
@Component
public class User{
}
當(dāng)用于Spring Boot應(yīng)用時(shí),被注解的類必須在啟動(dòng)類的根路徑或者子路徑下,否則不會(huì)生效。
如果不在,可以使用@ComponentScan標(biāo)注掃描的路徑。
spring xml 也有相關(guān)的標(biāo)簽<component-scan />
@ComponentScan(value={"com.microblog.blog","com.microblog.common"})
public class MicroblogBlogApplication {
public static void main(String args[]){
SpringApplication.run(MicroblogBlogApplication.class,args);
}
}
方式3:
使用@Bean注解,這種方式用在Spring Boot 應(yīng)用中。
@Configuration 標(biāo)識(shí)這是一個(gè)Spring Boot 配置類,其將會(huì)掃描該類中是否存在@Bean 注解的方法,比如如下代碼,將會(huì)創(chuàng)建User對(duì)象并放入容器中。
@ConditionalOnBean 用于判斷存在某個(gè)Bean時(shí)才會(huì)創(chuàng)建User Bean.
這里創(chuàng)建的Bean名稱默認(rèn)為方法的名稱user。也可以@Bean("xxxx")定義。
@Configuration
public class UserConfiguration{
@Bean
@ConditionalOnBean(Location.class)
public User user(){
return new User();
}
}
Spring boot 還為我們提供了更多類似的注解。

也和方式2一樣,也會(huì)存在掃描路徑的問(wèn)題,除了以上的解決方式,還有使用Spring boot starter 的解決方式
在resources下創(chuàng)建如下文件。META-INF/spring.factories.
Spring Boot 在啟動(dòng)的時(shí)候?qū)?huì)掃描該文件,從何獲取到配置類UserConfiguration。

spring.factories.
org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.log.config.UserConfiguration
如果不成功,請(qǐng)引入該依賴
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-configuration-processor</artifactId> <optional>true</optional> </dependency>
這個(gè)方式也是創(chuàng)建SpringBoot-starter的方式。
方式4:
使用注解@Import,也會(huì)創(chuàng)建對(duì)象并注入容器中
@Import(User.class)
public class MicroblogUserWebApplication {
public static void main(String args[]) {
SpringApplication.run(MicroblogUserWebApplication.class, args);
}
}
方式5:
使用ImportSelector或者ImportBeanDefinitionRegistrar接口,配合@Import實(shí)現(xiàn)。
在使用一些Spring Boot第三方組件時(shí),經(jīng)常會(huì)看到@EnableXXX來(lái)使能相關(guān)的服務(wù),這里以一個(gè)例子來(lái)實(shí)現(xiàn)。
創(chuàng)建測(cè)試類
@Slf4j
public class House {
public void run(){
log.info("House run ....");
}
}
@Slf4j
public class User {
public void run(){
log.info("User run ....");
}
}
@Slf4j
public class Student {
public void run(){
log.info("Student run ....");
}
}
實(shí)現(xiàn)ImportSelector接口
selectImports方法的返回值為需要?jiǎng)?chuàng)建Bean的類名稱。這里創(chuàng)建User類。
@Slf4j
public class MyImportSelector implements ImportSelector {
@Override
public String[] selectImports(AnnotationMetadata annotationMetadata) {
log.info("MyImportSelector selectImports ...");
return new String[]{
User.class.getName()};
}
}
實(shí)現(xiàn)ImportBeanDefinitionRegistrar接口
beanDefinitionRegistry.registerBeanDefinition用于設(shè)置需要?jiǎng)?chuàng)建Bean的類名稱。這里創(chuàng)建House類。
@Slf4j
public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {
@Override
public void registerBeanDefinitions(AnnotationMetadata annotationMetadata, BeanDefinitionRegistry beanDefinitionRegistry) {
log.info("MyImportBeanDefinitionRegistrar registerBeanDefinitions .....");
BeanDefinition beanDefinition = new RootBeanDefinition(House.class.getName());
beanDefinitionRegistry.registerBeanDefinition(House.class.getName(),beanDefinition);
}
}
創(chuàng)建一個(gè)配置類
這里創(chuàng)建Student類。
@Configuration
public class ImportAutoconfiguration {
@Bean
public Student student(){
return new Student();
}
}
創(chuàng)建EnableImportSelector注解
EnableImportSelector注解上使用@Import,引入以上的三個(gè)類。
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Target(ElementType.TYPE)
@Import({MyImportSelector.class,ImportAutoconfiguration.class,MyImportBeanDefinitionRegistrar.class})
public @interface EnableImportSelector {
String value();
}
測(cè)試
@EnableImportSelector(value = "xxx")
@SpringBootApplication
public class ImportDemoApplication {
public static void main(String[] args) {
ConfigurableApplicationContext context = SpringApplication.run(ImportDemoApplication.class, args);
User user = context.getBean(User.class);
user.run();
Student student = context.getBean(Student.class);
student.run();
House house = context.getBean(House.class);
house.run();
}
}
輸出,可以看到,三個(gè)類User Student House都創(chuàng)建成功,都可從Spring 容器中獲取到。
2019-06-20 17:53:39.528 INFO 27255 --- [ main] com.springboot.importselector.pojo.User : User run .... 2019-06-20 17:53:39.530 INFO 27255 --- [ main] c.s.importselector.pojo.Student : Student run .... 2019-06-20 17:53:39.531 INFO 27255 --- [ main] c.springboot.importselector.pojo.House : House run ....
方式6
手動(dòng)注入Bean容器,有些場(chǎng)景下需要代碼動(dòng)態(tài)注入,以上方式都不適用。這時(shí)就需要?jiǎng)?chuàng)建 對(duì)象手動(dòng)注入。
通過(guò)DefaultListableBeanFactory注入。
registerSingleton(String beanName,Object object);
這里手動(dòng)使用new創(chuàng)建了一個(gè)Location對(duì)象。并注入容器中。
@Component
public class LocationRegister implements BeanFactoryAware {
@Override
public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
DefaultListableBeanFactory listableBeanFactory = (DefaultListableBeanFactory)beanFactory;
Location location = new Location();
listableBeanFactory.registerSingleton("location1",location);
}
}
這種方式的應(yīng)用場(chǎng)景是為接口創(chuàng)建動(dòng)態(tài)代理對(duì)象,并向SPRING容器注冊(cè)。
比如MyBatis中的Mapper接口,Mapper沒(méi)有實(shí)現(xiàn)類,啟動(dòng)時(shí)創(chuàng)建動(dòng)態(tài)代理對(duì)象,將該對(duì)象注冊(cè)到容器中,使用時(shí)只要@Autowired注入即可使用,調(diào)用接口方法將會(huì)被代理攔截,進(jìn)而調(diào)用相關(guān)的SqlSession執(zhí)行相關(guān)的SQL業(yè)務(wù)邏輯。
可以看以下它的繼承體系
DefaultListableBeanFactory 是ConfigurableListableBeanFactory的實(shí)現(xiàn)類。是對(duì)BeanFactory功能的擴(kuò)展。

測(cè)試代碼和以上一樣
Location location = context.getBean(Location.class); location.run();
本文的相關(guān)代碼位于 實(shí)例代碼 測(cè)試類
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Spring超詳細(xì)講解創(chuàng)建BeanDefinition流程
- SpringBoot詳細(xì)講解如何創(chuàng)建及刷新Spring容器bean
- Spring-Bean創(chuàng)建對(duì)象的步驟方式詳解
- Spring BPP中如何優(yōu)雅的創(chuàng)建動(dòng)態(tài)代理Bean詳解
- Spring工廠方法創(chuàng)建(實(shí)例化)bean實(shí)例代碼
- 詳解Spring Boot 使用Java代碼創(chuàng)建Bean并注冊(cè)到Spring中
- Spring創(chuàng)建Bean完成后執(zhí)行指定代碼的幾種實(shí)現(xiàn)方式
相關(guān)文章
Spring中的接口重試機(jī)制spring-retry之listeners參數(shù)解析
這篇文章主要介紹了Spring中的接口重試機(jī)制spring-retry之listeners參數(shù)解析,注解@Retryable有一個(gè)參數(shù)listeners沒(méi)有說(shuō)明,那么本篇文章我們?cè)敿?xì)介紹一個(gè)這個(gè)參數(shù)的用,需要的朋友可以參考下2024-01-01
springboot+element-ui實(shí)現(xiàn)多文件一次上傳功能
這篇文章主要介紹了springboot+element-ui多文件一次上傳功能,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-06-06
聊聊Spring?Cloud?Gateway過(guò)濾器精確控制異常返回問(wèn)題
這篇文章主要介紹了Spring?Cloud?Gateway過(guò)濾器精確控制異常返回問(wèn)題,本篇任務(wù)就是分析上述現(xiàn)象的原因,通過(guò)閱讀源碼搞清楚返回碼和響應(yīng)body生成的具體邏輯,需要的朋友可以參考下2021-11-11
Spring Boot郵箱鏈接注冊(cè)驗(yàn)證及注冊(cè)流程
這篇文章給大家介紹Spring Boot郵箱鏈接注冊(cè)驗(yàn)證問(wèn)題及注冊(cè)流程分析,通過(guò)實(shí)例代碼給大家分享實(shí)現(xiàn)過(guò)程,感興趣的朋友跟隨小編一起看看吧2021-07-07
spring?boot?3使用?elasticsearch?提供搜索建議的實(shí)例詳解
這篇文章主要介紹了spring?boot3使用elasticsearch提供搜索建議,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-08-08
詳解SpringBoot時(shí)間參數(shù)處理完整解決方案
這篇文章主要介紹了詳解SpringBoot時(shí)間參數(shù)處理完整解決方案,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12

