淺談Spring注入模型
Spring注入bean的方式
DI exists in two major variants: Constructor-based dependency injection and Setter-based dependency injection.
從Spring的官網(wǎng)中可以得知,Spring注入bean的方式由兩種,一種是通過構(gòu)造方法進(jìn)行注入,另外一種是通過setter方法進(jìn)行注入。
但熟悉Spring開發(fā)的同學(xué)在實(shí)際的開發(fā)中經(jīng)常使用的注入方式是通過@Autowired以及@Resource的方式來注入bean的。那么通過注解的方式注入是跟上面兩種注入方式是一樣的嗎?在研究這個(gè)問題之前,我們需要來了解Spring的注入模型。
Spring的注入模型
如果不深入到Spring的源碼,是很少有機(jī)會(huì)了解到Spring的注入模型(AutowireMode)。但是為了掃清我們學(xué)習(xí)Spring源碼的障礙,我們有必要了解下Spring的注入模型。
Spring的注入模型有四種,分別是:
- autowire_no(0):默認(rèn)的注入模型,如果在beanA中注入beanB,如果沒有提供注解或者xml的注入方式,beanB是無法注入到beanA中的。
- autowire_name(1):通過set方法注入,并且set的參數(shù)名必須和注入的bean名稱一樣,在xml中是通過byName
- autowire_type(2):通過set方法注入,set的參數(shù)名可以隨意命名,但是類型必須和注入的bean的類型一樣,在xml中是通過byType的方式配置
- autowire_constructor(3):通過構(gòu)造器注入
代碼示例
多說無益,我們直接上代碼來加深對(duì)這一概念的了解。
定義bean對(duì)象
我們簡(jiǎn)單定義兩個(gè)類,分別是Student和Address,并且將他們注入到Spring容器中。
Student.java如下
@Component @Slf4j public class Student { private Address address; public Student(){ log.info("default constructor..."); } public Student(Address address){ log.info("constructor inject...{}", address); this.address = address; } public void setAddress(Address address){ log.info("setter...{}", address); this.address = address; } public void showAddress(){ this.address.info(); } }
Address.java如下
@Component public class Address { public void info(){ System.out.println("廣東省廣州市白云區(qū)"); } }
我們可以看到在Student中有一個(gè)屬性是Address,并且提供了默認(rèn)的構(gòu)造方法以及帶參數(shù)的構(gòu)造方法,同時(shí)還有一個(gè) set方法。
定義配置類
我們?cè)俣x一個(gè)config類,用來掃描這兩個(gè)類所在的包路徑,把這兩個(gè)類注入到Spring容器中。
@ComponentScan("org.example.autoModel.model") public class MyBeanConfig { }
定義后置處理器
在Spring中,我們可以在自己定義的后置處理器BeanFactoryPostProcessor,獲取到對(duì)應(yīng)的BeanDefinition。
相信大家在學(xué)習(xí)Spring的時(shí)候有了解過BeanDefinition,不熟悉的可以搜索下BeanDefinition的概念,如果閱讀過Spring的源碼的話,對(duì)BeanDefinition應(yīng)該就更加熟悉了。這里就不做過多的解釋了,后面在解讀Spring源碼的時(shí)候應(yīng)該也會(huì)涉及到這部分的知識(shí)。
簡(jiǎn)單的來說,就是一個(gè)對(duì)象被注入到Spring中,實(shí)際是被解析成BeanDefinition對(duì)象,里面保存了各種需要用到的信息,并且可以在后置處理器中獲取到對(duì)應(yīng)的BeanDefinition對(duì)象,然后對(duì)其做一些操作。
@Slf4j(topic = "e") public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor { @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { AbstractBeanDefinition beanDefinition = (AbstractBeanDefinition) beanFactory.getBeanDefinition("student"); // beanDefinition.setAutowireMode(3); log.debug("mode:{}",beanDefinition.getAutowireMode()); } }
上面代碼所示,我們可以獲取這個(gè)beanDefinition在Spring容器中對(duì)應(yīng)的autowireMode的值,并且可以修改這個(gè)autowireMode的值,來觀察下注入模型的改變,對(duì)bean的注入方式有什么影響。
定義測(cè)試方法
@Test public void modelTest(){ AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); context.register(MyBeanConfig.class); context.register(MyBeanFactoryPostProcessor.class); context.refresh(); Student student = context.getBean(Student.class); student.showAddress(); }
執(zhí)行測(cè)試方法,得出如下的結(jié)果。
如上面所示,我們可以得到對(duì)應(yīng)的autowireMode的值是0,并且是執(zhí)行了Student的默認(rèn)構(gòu)造方法,由于沒有將address屬性注入到student中,所在調(diào)用address對(duì)應(yīng)的方法的時(shí)候,拋出了空指針的異常。
針對(duì)上面的對(duì)象,我們可以簡(jiǎn)單分析下:
- Spring默認(rèn)的注入模型是0
- 注入模型是0,意味著執(zhí)行默認(rèn)的構(gòu)造方法,并且不會(huì)執(zhí)行set方法去進(jìn)行注入address這個(gè)對(duì)象,所以拋出了空指針異常
我們可以在后置處理器中,修改對(duì)應(yīng)autowioreMode的值,將autowireMode的值修改成1。
beanDefinition.setAutowireMode(1);
然后執(zhí)行該測(cè)試方法,得到下面的結(jié)果。
可以看到我們將注入模型的值修改成1,之后,spring就會(huì)通過set方法,將address注入到student中,就可以成功調(diào)用address的方法了。
當(dāng)然,我們將注入模型的值修改成2,也是注入成功的,并且也是通過set方法,只不過是byType和byName的區(qū)別。這個(gè)同學(xué)們可以自己嘗試下。
下面,我們將注入模型的值修改成3來看下執(zhí)行的結(jié)果。
beanDefinition.setAutowireMode(3);
結(jié)果如下:
可以看到,如果將注入模型的值修改成了3,那么Spring是通過帶參的構(gòu)造方法來注入給student的。
通過上面簡(jiǎn)單的例子,我們就可以了解到注入模型autowireMode對(duì)注入方式的影響,這為我們以后閱讀Spring的源碼打下了基礎(chǔ)。
自動(dòng)注入和手動(dòng)注入
上面我們將autowireMode值修改成1、2、3,就可以完成屬性的自動(dòng)注入。那如果我們不去修改autowireMode的值,而是使用注解來將屬性注入到student中,會(huì)是怎樣的結(jié)果呢?
我們來看下面的代碼:
@Component @Slf4j public class Student { @Autowired private Address address; public Student(){ log.info("default constructor..."); } public Student(Address address){ log.info("constructor inject...{}", address); this.address = address; } public void setAddress(Address address){ log.info("setter...{}", address); this.address = address; } public void showAddress(){ this.address.info(); } }
@Slf4j(topic = "e") public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor { @Override public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { AbstractBeanDefinition beanDefinition = (AbstractBeanDefinition) beanFactory.getBeanDefinition("student"); // beanDefinition.setAutowireMode(3); log.debug("mode:{}",beanDefinition.getAutowireMode()); } }
查看結(jié)果
我們發(fā)現(xiàn)使用@Autowired注解注入屬性的時(shí)候,注入模型的值是0,也就是默認(rèn)的值。
對(duì)于@Autowired的注解,我的看法是,使用注解注入屬性是手動(dòng)注入的方式,Spring提供自動(dòng)注入的方式只有set方法以及帶參的構(gòu)造方法。而注解的注入方式,只不過是xml手動(dòng)注入的一種簡(jiǎn)化,Spring內(nèi)部處理的機(jī)制都是類似的。
了解過xml配置的同學(xué)大概有些印象,我們需要指定bean中屬性所對(duì)應(yīng)的類的全路徑。而使用自動(dòng)注入,我們可以指定byType或者byName。由于xml方式太久沒用了,大概就是這么個(gè)意思吧。
所以我認(rèn)為@Autowired注入的方式,不是自動(dòng)注入,而是手動(dòng)注入,只不過Spring容器內(nèi)部幫我們處理了。當(dāng)然,關(guān)于Spring容器如何處理@Autowired注解的方式注入屬性的,同學(xué)們可以通過閱讀源碼來大致了解下。后面大概也會(huì)解讀下這部分的源碼。
以上便是鄙人對(duì)Spring注入模型的淺薄認(rèn)識(shí),如果有錯(cuò)誤的話,還請(qǐng)同學(xué)們多多包涵。
到此這篇關(guān)于淺談Spring注入模型的文章就介紹到這了,更多相關(guān)Spring注入模型內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java?中?Class?Path?和?Package的使用詳解
這篇文章主要介紹了Java?中?Class?Path和Package的使用詳解,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的朋友可以參考一下2022-08-08java開發(fā)ServiceLoader實(shí)現(xiàn)機(jī)制及SPI應(yīng)用
這篇文章主要為大家介紹了java開發(fā)ServiceLoader實(shí)現(xiàn)機(jī)制及SPI應(yīng)用,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10Java類如何實(shí)現(xiàn)一個(gè)類的障眼法(JadClipse的bug)
這篇文章主要介紹了Java類實(shí)現(xiàn)一個(gè)類的障眼法(JadClipse的bug),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12Java對(duì)象初始化過程代碼塊和構(gòu)造器的調(diào)用順序
這篇文章主要介紹了Java對(duì)象初始化過程代碼塊和構(gòu)造器的調(diào)用順序,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-08-08關(guān)于mybatis-plus-generator的簡(jiǎn)單使用示例詳解
在springboot項(xiàng)目中集成mybatis-plus是很方便開發(fā)的,最近看了一下plus的文檔,簡(jiǎn)單用一下它的代碼生成器,接下來通過實(shí)例代碼講解關(guān)于mybatis-plus-generator的簡(jiǎn)單使用,感興趣的朋友跟隨小編一起看看吧2024-03-03Java數(shù)字格式類(NumberFormat類和DecimalFormat類)用法詳解
NumberFormat類是Java提供的一個(gè)格式化數(shù)字的類,可以將一串?dāng)?shù)字轉(zhuǎn)化成自己想要的數(shù)據(jù)格式,也可以將字符串轉(zhuǎn)化成數(shù)值,下面這篇文章主要給大家介紹了關(guān)于Java數(shù)字格式類(NumberFormat類和DecimalFormat類)用法的相關(guān)資料,需要的朋友可以參考下2022-07-07java實(shí)現(xiàn)數(shù)據(jù)結(jié)構(gòu)單鏈表示例(java單鏈表)
這篇文章主要介紹了java數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)單鏈表示例,需要的朋友可以參考下2014-03-03