Java面試題沖刺第十八天--Spring框架3
面試題1:Bean 的加載過程是怎樣的?
我們知道, Spring 的工作流主要包括以下兩個環(huán)節(jié):
- 解析,讀 xml 配置,掃描類文件,從配置或者注解中獲取 Bean 的定義信息,注冊一些擴展功能。
- 加載,通過解析完的定義信息獲取 Bean 實例。
下面是跟蹤了 getBean的調(diào)用鏈創(chuàng)建的流程圖,為了能夠很好地理解 Bean 加載流程,省略一些異常、日志和分支處理和一些特殊條件的判斷。

從上面的流程圖中,可以看到一個 Bean 加載主要會經(jīng)歷這么幾個階段(標綠內(nèi)容):
- 獲取 BeanName,對傳入的 name 進行解析,轉(zhuǎn)化為可以從 Map 中獲取到 BeanDefinition 的 bean name。
- 合并 Bean 定義,對父類的定義進行合并和覆蓋,如果父類還有父類,會進行遞歸合并,以獲取完整的 Bean 定義信息。
- 實例化,使用構(gòu)造或者工廠方法創(chuàng)建 Bean 實例。
- 屬性填充,尋找并且注入依賴,依賴的 Bean 還會遞歸調(diào)用 getBean 方法獲取。
- 初始化,調(diào)用自定義的初始化方法。
- 獲取最終的 Bean,如果是 FactoryBean 需要調(diào)用 getObject 方法,如果需要類型轉(zhuǎn)換調(diào)用 TypeConverter 進行轉(zhuǎn)化。
以上便是Spring對bean解析注冊的全過程,總結(jié)一下大致步驟:
- 加載XML文件,封裝成Resource對象;
- 調(diào)用Reader對象方法讀取XML文件內(nèi)容,并將相關(guān)屬性放到BeanDefinition實例;
- 將BeanDefinition對象放到BeanFactory對象,用于調(diào)用;
追問1:什么是循環(huán)依賴?
舉個例子,這里有三個類 A、B、C,然后 A 關(guān)聯(lián) B,B 關(guān)聯(lián) C,C 又關(guān)聯(lián) A,這就形成了一個循環(huán)依賴。如果是方法調(diào)用是不算循環(huán)依賴的,循環(huán)依賴必須要持有引用。

循環(huán)依賴發(fā)生的場景:
- 構(gòu)造器循環(huán)依賴:依賴的對象是通過構(gòu)造器傳入的,發(fā)生在實例化 Bean 的時候。
- 設(shè)值循環(huán)依賴:依賴的對象是通過 setter 方法傳入的,對象已經(jīng)實例化,發(fā)生屬性填充和依賴注入的時候。
- 如果是構(gòu)造器循環(huán)依賴,本質(zhì)上是無法解決的。比如我們準調(diào)用 A 的構(gòu)造器,發(fā)現(xiàn)依賴 B,于是去調(diào)用 B 的構(gòu)造器進行實例化,發(fā)現(xiàn)又依賴 C,于是調(diào)用 C 的構(gòu)造器去初始化,結(jié)果依賴 A,整個形成一個死結(jié),導致 A 無法創(chuàng)建。
- 如果是設(shè)值循環(huán)依賴,Spring 框架只支持單例下的設(shè)值循環(huán)依賴。Spring 通過對還在創(chuàng)建過程中的單例,緩存并提前暴露該單例,使得其他實例可以引用該依賴。
追問2:循環(huán)依賴得解決思路是什么樣的?
Spring解決循環(huán)依賴,主要的思路就是依據(jù)三級緩存(解鏈)。
在實例化A時調(diào)用doGetBean,發(fā)現(xiàn)A依賴的B的實例,此時調(diào)用doGetBean去實例B,實例化的B的時候發(fā)現(xiàn)又依賴A,如果不解決這個循環(huán)依賴的話此時的doGetBean將會無限循環(huán)下去,導致內(nèi)存溢出,程序奔潰。
如果Spring引用一個早期對象,并且把這個"早期引用"并將其注入到容器中,讓B先完成實例化,此時A就獲取B的引用,完成實例化。
一級緩存:singletonObjects,存放完全實例化屬性賦值完成的Bean,直接可以使用。
二級緩存:earlySingletonObjects,存放早期Bean的引用,尚未屬性裝配的Bean
三級緩存:singletonFactories,三級緩存,存放實例化完成的Bean工廠。
面試題2:@Resource和@Autowired有什么區(qū)別?
- @Autowired 根據(jù)類型注入
- @Resource 默認根據(jù)名字注入,其次按照類型搜索
- @Autowired @Qualifie("userService") 兩個結(jié)合起來可以根據(jù)名字和類型注入,等同于@Resource
1.@Autowired與@Resource都可以用來裝配bean. 都可以寫在字段上,或?qū)懺趕etter方法上。
2.@Autowired默認按類型裝配(byType),默認情況下必須要求依賴對象必須存在,如果要允許null值,可以設(shè)置它的required屬性為false,如:@Autowired(required=false) ,如果我們想使用名稱裝配可以結(jié)合@Qualifier注解進行使用(@Autowired () @Qualifier ( "xxx" )功能同@Resource),如下:
@Autowired @Qualifier ( "userDao" ) private UserDao userDao;
3.@Resource默認按照名稱進行裝配(byName),名稱可以通過name屬性進行指定,如果沒有指定name屬性,當注解寫在字段上時,默認取字段名進行安裝名稱查找,如果注解寫在setter方法上默認取屬性名進行裝配。當找不到與名稱匹配的bean時才按照類型進行裝配。如果name屬性一旦指定,就只會按照名稱進行裝配。
@Resource (name= "baseDao" ) private BaseDao baseDao;
總結(jié)如下:
- @Autowired默認按byType自動裝配,而@Resource默認byName自動裝配。
- @Autowired只包含一個參數(shù):required,表示是否開啟自動注入,默認是true。而@Resource包含七個參數(shù),其中最重要的兩個參數(shù)是:name 和 type。
- @Autowired如果要使用byName,需要使用@Qualifier一起配合。而@Resource如果指定了name,則用byName自動裝配,如果指定了type,則用byType自動裝配。
- @Autowired能夠用在:構(gòu)造器、方法、參數(shù)、成員變量和注解上,而@Resource能用在:類、成員變量和方法上。
- @Autowired是spring定義的注解,而@Resource是JSR-250定義的注解。
面試題3:Spring 的事務傳播行為有哪些,都有什么作用?
簡單來講,就是當系統(tǒng)中存在兩個事務方法時(我們暫稱為方法A和方法B),如果方法B在方法A中被調(diào)用,那么將采用什么樣的事務形式,就叫做事務的傳播特性
比如,A方法調(diào)用了B方法(B方法必須使用事務注解),那么B事務可以是一個在A中嵌套的事務,或者B事務不使用事務,又或是使用與A事務相同的事務,這些均可以通過指定事務傳播特性來實現(xiàn)。
| 傳播行為 | 意義 |
|---|---|
| propagation.REQUIRED | 表示當前方法必須運行在事務中。如果當前事務存在,方法將會在該事務中運行。否則會啟動一個新的事務 |
| propagation.SUPPORTS | 表示當前方法不需要事務上下文,但是如果存在當前事務的話,那么該方法會在這個事務中運行 |
| propagation.MANDATORY | 表示該方法必須在事務中運行,如果當前事務不存在,則會拋出一個異常 |
| propagation.REQUIRED_NEW | 表示當前方法必須運行在它自己的事務中。一個新的事務將被啟動。如果存在當前事務,在該方法執(zhí)行期間,當前事務會被掛起。如果使用JTATransactionManager的話,則需要訪問TransactionManager |
| propagation.NOT_SUPPORTED | 表示該方法不應該運行在事務中。如果存在當前事務,在該方法運行期間,當前事務將被掛起。如果使用JTATransactionManager的話,則需要訪問TransactionManager |
| propagation.NEVER | 表示當前方法不應該運行在事務上下文中。如果當前正有一個事務在運行,則會拋出異常 |
| propagation.NESTED | 表示如果當前已經(jīng)存在一個事務,那么該方法將會在嵌套事務中運行。嵌套的事務可以獨立于當前事務進行單獨地提交或回滾。如果當前事務不存在,那么其行為與propagation.REQUIRED一樣。注意各廠商對這種傳播行為的支持是有所差異的??梢詤⒖假Y源管理器的文檔來確認它們是否支持嵌套事務 |
總結(jié)
本篇文章就到這里了,希望能給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
SpringBoot如何使用TestEntityManager進行JPA集成測試
TestEntityManager是Spring Framework提供的一個測試框架,它可以幫助我們進行 JPA 集成測試,在本文中,我們將介紹如何使用 TestEntityManager 進行 JPA 集成測試,感興趣的跟著小編一起來學習吧2023-06-06
Java 回調(diào)機制(CallBack) 詳解及實例代碼
這篇文章主要介紹了 Java 回調(diào)機制(CallBack) 詳解及實例代碼的相關(guān)資料,需要的朋友可以參考下2017-02-02
Springboot Thymeleaf實現(xiàn)HTML屬性設(shè)置
這篇文章主要介紹了Springboot Thymeleaf實現(xiàn)HTML屬性設(shè)置,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2007-11-11
jquery uploadify和apache Fileupload實現(xiàn)異步上傳文件示例
這篇文章主要介紹了jquery uploadify和apache Fileupload實現(xiàn)異步上傳文件示例,需要的朋友可以參考下2014-05-05
springboot用controller跳轉(zhuǎn)html頁面的實現(xiàn)
這篇文章主要介紹了springboot用controller跳轉(zhuǎn)html頁面的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-09-09

