spring @Lazy延遲注入的邏輯實(shí)現(xiàn)
前言
有時(shí)候我們會(huì)在屬性注入的時(shí)候添加@Lazy注解實(shí)現(xiàn)延遲注入,今天咱們通過(guò)閱讀源碼來(lái)分析下原因
一、一個(gè)簡(jiǎn)單的小例子
代碼如下:
@Service public class NormalService1 { @Autowired @Lazy private MyService myService; public void doSomething() { myService.getName(); } }
作用是為了進(jìn)行延遲加載,在NormalService1進(jìn)行屬性注入的時(shí)候,如果MyService還沒(méi)有生成bean也不用擔(dān)心,會(huì)注入一個(gè)代理,但是在實(shí)際運(yùn)行的時(shí)候,會(huì)獲取Spring容器中實(shí)際的MyService,在某些情況下,因?yàn)閟pring生命周期的原因,這個(gè)注解有大用。
二、源碼解讀
1. 注入
代碼如下(DefaultListableBeanFactory#resolveDependency):
public Object resolveDependency(DependencyDescriptor descriptor, @Nullable String requestingBeanName, @Nullable Set<String> autowiredBeanNames, @Nullable TypeConverter typeConverter) throws BeansException { descriptor.initParameterNameDiscovery(getParameterNameDiscoverer()); if (Optional.class == descriptor.getDependencyType()) { return createOptionalDependency(descriptor, requestingBeanName); } else if (ObjectFactory.class == descriptor.getDependencyType() || ObjectProvider.class == descriptor.getDependencyType()) { return new DependencyObjectProvider(descriptor, requestingBeanName); } else if (javaxInjectProviderClass == descriptor.getDependencyType()) { return new Jsr330Factory().createDependencyProvider(descriptor, requestingBeanName); } else { //如果注入屬性添加了@Lazy,懶加載,此時(shí)spring會(huì)根據(jù)具體類型搞個(gè)cglib代理類 Object result = getAutowireCandidateResolver().getLazyResolutionProxyIfNecessary( descriptor, requestingBeanName); if (result == null) { result = doResolveDependency(descriptor, requestingBeanName, autowiredBeanNames, typeConverter); } return result; } }
很明顯要執(zhí)行g(shù)etLazyResolutionProxyIfNecessary方法,如果加了@Lazy注解,最終會(huì)執(zhí)行buildLazyResolutionProxy方法
protected Object buildLazyResolutionProxy(final DependencyDescriptor descriptor, final @Nullable String beanName) { Assert.state(getBeanFactory() instanceof DefaultListableBeanFactory, "BeanFactory needs to be a DefaultListableBeanFactory"); final DefaultListableBeanFactory beanFactory = (DefaultListableBeanFactory) getBeanFactory(); TargetSource ts = new TargetSource() { @Override public Class<?> getTargetClass() { return descriptor.getDependencyType(); } @Override public boolean isStatic() { return false; } @Override public Object getTarget() { Object target = beanFactory.doResolveDependency(descriptor, beanName, null, null); /** something valid **/ return target; } @Override public void releaseTarget(Object target) { } }; ProxyFactory pf = new ProxyFactory(); pf.setTargetSource(ts); Class<?> dependencyType = descriptor.getDependencyType(); if (dependencyType.isInterface()) { pf.addInterface(dependencyType); } return pf.getProxy(beanFactory.getBeanClassLoader()); }
可以看到上面這段代碼,其實(shí)就是生成了一個(gè)TargetSource,然后再生成了一個(gè)代理(CGLIB或者JDK),然后作為MyService對(duì)象注入給了NormalService1。那么所謂的執(zhí)行的過(guò)程中才進(jìn)行獲取真正的MyService對(duì)象是什么意思呢?
2. 使用邏輯
本文示例代碼使用的是CGLIB代理,其實(shí)是類似的,因?yàn)樽⑷氲腗yService是個(gè)CGLIB代理對(duì)象,那么在執(zhí)行方法的時(shí)候,就會(huì)調(diào)用CglibAopProxy#DynamicAdvisedInterceptor#intercept方法
那么此處其實(shí)調(diào)用的就是上面的
Object target = beanFactory.doResolveDependency(descriptor, beanName, null, null);
這個(gè)方法就不用認(rèn)真看了,主要功能就是從Spring容器中找到MyService。
在之前講@Autowired原理和@Resource注入原理的時(shí)候解釋過(guò)了,不清楚的可以看專欄里其他文章。
拿出來(lái)之后會(huì)發(fā)現(xiàn),咱們拿到的target對(duì)象還是一個(gè)CGLIB增加的對(duì)象
那么當(dāng)執(zhí)行方法邏輯時(shí)
由于target是CGLIB對(duì)象,會(huì)再次進(jìn)入到CglibAopProxy#DynamicAdvisedInterceptor#intercept方法。
此時(shí)拿到的target對(duì)象類型就不同了
是我們代理之前的target對(duì)象,此時(shí)再次進(jìn)行invoke的時(shí)候,就會(huì)進(jìn)行動(dòng)態(tài)代理的一般邏輯,先查找該方法匹配的所有advice,然后依次調(diào)用,最終調(diào)用target本身對(duì)于方法的執(zhí)行。
總結(jié)
所以可以發(fā)現(xiàn)其實(shí)@Lazy只不過(guò)是給spring的代理對(duì)象proxy再進(jìn)行了一次proxy,只不過(guò)沒(méi)有在注入的時(shí)候,就獲取到對(duì)象,而是借用了方法invoke時(shí)通過(guò)proxy的intercept方法getTarget,然后進(jìn)行方法調(diào)用,延遲了對(duì)象的注入。之后每次調(diào)用的時(shí)候都需要從Spring容器中獲取到原生的proxy對(duì)象。
到此這篇關(guān)于spring @Lazy延遲注入的邏輯實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)spring @Lazy延遲注入內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Mybatis-Plus3.2.0 MetaObjectHandler 無(wú)法進(jìn)行公共字段全局填充
這篇文章主要介紹了Mybatis-Plus3.2.0 MetaObjectHandler 無(wú)法進(jìn)行公共字段全局填充,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-11-11Mybatis-Plus?sum聚合函數(shù)及按日期查詢并求和的方式詳解
這篇文章主要介紹了Mybatis-Plus sum聚合函數(shù)及按日期查詢并求和,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-06-06解決SpringBoot使用@Value獲取不到y(tǒng)aml中配置值的問(wèn)題
在最近的開(kāi)發(fā)中遇到一個(gè)問(wèn)題,使用@Value獲取yml文件中配置的屬性時(shí)始終獲取不到值,所以本文給大家詳細(xì)介紹了SpringBoot使用@Value獲取不到y(tǒng)aml中值的問(wèn)題分析及解決方法,需要的朋友可以參考下2024-01-01Java SpringBoot微服務(wù)框架驗(yàn)證碼報(bào)錯(cuò)問(wèn)題解決方案
這篇文章主要介紹了Java SpringBoot微服務(wù)框架驗(yàn)證碼報(bào)錯(cuò)問(wèn)題解決方案,包括dockerfile容器操作和完整dockerfile,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下2024-08-08java.net.UnknownHostException異常的一般原因及解決步驟
關(guān)于java.net.UnknownHostException大家也許都比較熟悉,這篇文章主要給大家介紹了關(guān)于java.net.UnknownHostException異常的一般原因及解決步驟,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-02-02Mapper層繼承BaseMapper<T>需要引入的pom依賴方式
這篇文章主要介紹了Mapper層繼承BaseMapper<T>需要引入的pom依賴方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01java計(jì)算給定字符串中出現(xiàn)次數(shù)最多的字母和該字母出現(xiàn)次數(shù)的方法
這篇文章主要介紹了java計(jì)算給定字符串中出現(xiàn)次數(shù)最多的字母和該字母出現(xiàn)次數(shù)的方法,涉及java字符串的遍歷、轉(zhuǎn)換及運(yùn)算相關(guān)操作技巧,需要的朋友可以參考下2017-02-02Hibernatede 一對(duì)多映射配置方法(分享)
下面小編就為大家?guī)?lái)一篇Hibernatede 一對(duì)多映射配置方法(分享)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-09-09