spring getBean 源碼學(xué)習(xí)(下)

上一篇spring getBean 源碼學(xué)習(xí)(上)基本上把getBean的過程細(xì)節(jié)了解清楚了,還剩下一些疑問以及沒有注意到的細(xì)節(jié)昼弟,本篇文章就來深入細(xì)節(jié)饲握,解析之前遺留的問題救欧,最好是配合著上篇一起看。

現(xiàn)在已知的存在不同類型(scope)的bean被存儲在容器中

并不存在這樣的铝耻,spring整個的bean管理容器是DefaultListableBeanFactory,他有很多個線程安全或者線程不安全的容器存有不同類型的數(shù)據(jù)蹬刷,可是具體bean的信息是存儲在BeanDefinition中的办成,在AbstractBeanDefinition中有個scope的數(shù)據(jù),他就存儲著bean的具體類型某弦,在使用的時候直接判斷其scope即可

把獲取的bean列表 循環(huán)一遍依次實例化靶壮、填充數(shù)據(jù)

是這樣操作的员萍,在解析xml的時候,finishBeanFactoryInitialization方法會實例化所有還沒處理的single對象

有一個問題不知道有沒有想到螃壤,實際開發(fā)中很少有使用getBean操作去獲取相應(yīng)bean映穗,那么那些注解依賴的bean是如何生成的呢?下面是我們使用的demo聊一聊這個問題

image.png
image.png

Teacher類加上了Component這個注解,spring啟動的時候就會解析實例化這個類睦霎。
原因就在下面這個xml配置上

image.png

spring xml的bean提取 源碼學(xué)習(xí)文章中對NameSpace進(jìn)行了詳細(xì)的解釋副女。在此不做過多說明蚣旱,直接看圖

image.png

很明顯了塞绿,相關(guān)bean的注冊操作就是依靠ComponentScanBeanDefinitionParser類完成异吻,xml文件掃描完成之后,在refresh中的finishBeanFactoryInitialization完成實例化操作棋返。如下代碼完成了bean的注冊工作雷猪。

ClassPathBeanDefinitionScanner 文件

protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
    Assert.notEmpty(basePackages, "At least one base package must be specified");
    Set<BeanDefinitionHolder> beanDefinitions = new LinkedHashSet<BeanDefinitionHolder>();
    for (String basePackage : basePackages) {
        Set<BeanDefinition> candidates = findCandidateComponents(basePackage);
        for (BeanDefinition candidate : candidates) {
            ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(candidate);
            candidate.setScope(scopeMetadata.getScopeName());
            String beanName = this.beanNameGenerator.generateBeanName(candidate, this.registry);
            if (candidate instanceof AbstractBeanDefinition) {
                postProcessBeanDefinition((AbstractBeanDefinition) candidate, beanName);
            }
            if (candidate instanceof AnnotatedBeanDefinition) {
                AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition) candidate);
            }
            if (checkCandidate(beanName, candidate)) {
                BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(candidate, beanName);
                definitionHolder =
                        AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
                beanDefinitions.add(definitionHolder);
                registerBeanDefinition(definitionHolder, this.registry);
            }
        }
    }
    return beanDefinitions;
}

各種不同類型的容器酵颁,例如實例化好的月帝,正在實例化的嚷辅、正在被銷毀的距误、已經(jīng)銷毀的

如下代碼,在DefaultListableBeanFactory文件中有多種容器趁俊,以完成各種需求

// 工廠的序列號id寺擂,唯一性
private String serializationId;

// 允許bean出現(xiàn)覆蓋的情況
private boolean allowBeanDefinitionOverriding = true;

// 是否提前加載,而不需要關(guān)系是否存在懶加載的情況
private boolean allowEagerClassLoading = true;

// 針對依賴列表的選擇器
private Comparator<Object> dependencyComparator;

// 目前還沒搞懂具體的含義
private AutowireCandidateResolver autowireCandidateResolver = new SimpleAutowireCandidateResolver();

// 依賴對應(yīng)的具體對象
private final Map<Class<?>, Object> resolvableDependencies = new ConcurrentHashMap<Class<?>, Object>(16);

// 存儲bean的容器垦细,以name為key
private final Map<String, BeanDefinition> beanDefinitionMap = new ConcurrentHashMap<String, BeanDefinition>(256);

// 以className為key的線程安全的容器括改,包括了single和非single(prototype)
private final Map<Class<?>, String[]> allBeanNamesByType = new ConcurrentHashMap<Class<?>, String[]>(64);

// 單例single的容器家坎,以className為key
private final Map<Class<?>, String[]> singletonBeanNamesByType = new ConcurrentHashMap<Class<?>, String[]>(64);

// beanName 為集合的list
private volatile List<String> beanDefinitionNames = new ArrayList<String>(256);
...

其中關(guān)鍵字allowBeanDefinitionOverriding很有意思虱疏,是關(guān)于重名bean的订框,可以查看spring 同名bean問題 源碼學(xué)習(xí) 具體了解其中的用法

填充數(shù)據(jù)這部分操作得支持@value等類似操作

@value作為一個注解,在填充到bean中肯定需要從配置文件獲取其值衩侥,然后才可以設(shè)置類的字段意思矛物。接下來就具體學(xué)習(xí)下其中的原理

ComponentScanBeanDefinitionParser 類

這個類是為<context:component-scan base-package="com.demo"/>履羞,他會注入接下來需要使用的AutowiredAnnotationBeanPostProcessor

image.png

接下來峦萎,在填充值的時候就會利用到AutowiredAnnotationBeanPostProcessor類了。

image.png

AutowiredAnnotationBeanPostProcessor 文件

從populateBean函數(shù)進(jìn)入的忆首,設(shè)置屬性值


public PropertyValues postProcessPropertyValues(PropertyValues pvs,     
    PropertyDescriptor[] pds, Object bean, String beanName) 
        throws BeanCreationException {

    InjectionMetadata metadata = findAutowiringMetadata(beanName, bean.getClass(), pvs);
    // 從該AutowiredAnnotationBeanPostProcessor的cache中獲取該名稱的注入元類信息
    try {
        metadata.inject(bean, beanName, pvs);
        // 注入值
    }
    catch (BeanCreationException ex) {
        throw ex;
    }
    catch (Throwable ex) {
        throw new BeanCreationException(beanName, "Injection of autowired dependencies failed", ex);
    }
    return pvs;
}

InjectionMetadata 文件

public void inject(Object target, String beanName, PropertyValues pvs) 
                    throws Throwable {
    Collection<InjectedElement> elementsToIterate =
            (this.checkedElements != null ? this.checkedElements : this.injectedElements);
   // 獲取的元素一次循環(huán)進(jìn)行注入inject操作
    if (!elementsToIterate.isEmpty()) {
        boolean debug = logger.isDebugEnabled();
        for (InjectedElement element : elementsToIterate) {
            if (debug) {
                logger.debug("Processing injected element of bean '" + beanName + "': " + element);
            }
            element.inject(target, beanName, pvs);
        }
    }
}

AutowiredAnnotationBeanPostProcessor 文件

protected void inject(Object bean, String beanName, PropertyValues pvs) 
                   throws Throwable {
        Field field = (Field) this.member;
        Object value;
        if (this.cached) {
            value = resolvedCachedArgument(beanName, this.cachedFieldValue);
        }
        else {
            DependencyDescriptor desc = new DependencyDescriptor(field, this.required);
            desc.setContainingClass(bean.getClass());
            Set<String> autowiredBeanNames = new LinkedHashSet<String>(1);
            TypeConverter typeConverter = beanFactory.getTypeConverter();
            try {
                value = beanFactory.resolveDependency(desc, beanName, autowiredBeanNames, typeConverter);
                // 調(diào)用bean工廠DefaultListableBeanFactory 的解決依賴功能
                // 得到其真實的值
                // TODO 目前也沒看懂具體的細(xì)節(jié)爱榔,(/_\)
            }
            catch (BeansException ex) {
                throw new UnsatisfiedDependencyException(null, beanName, new InjectionPoint(field), ex);
            }
            synchronized (this) {
                if (!this.cached) {
                    if (value != null || this.required) {
                        this.cachedFieldValue = desc;
                        registerDependentBeans(beanName, autowiredBeanNames);
                        if (autowiredBeanNames.size() == 1) {
                            String autowiredBeanName = autowiredBeanNames.iterator().next();
                            if (beanFactory.containsBean(autowiredBeanName)) {
                                if (beanFactory.isTypeMatch(autowiredBeanName, field.getType())) {
                                    this.cachedFieldValue = new ShortcutDependencyDescriptor(
                                            desc, autowiredBeanName, field.getType());
                                }
                            }
                        }
                    }
                    else {
                        this.cachedFieldValue = null;
                    }
                    this.cached = true;
                }
            }
        }
        if (value != null) {
            ReflectionUtils.makeAccessible(field);
            // 最后直接修改這個field的屬性值
            field.set(bean, value);
        }
    }
}

總結(jié):其實沒有非常理解其在賦值的操作細(xì)節(jié),跨度太長糙及,不過也清楚了如果需要使用依賴注入的功能更详幽,則一定需要在xml中加入<context:component-scan base-package="com.demo"/>這樣類似的話。

對外提供各種接口唇聘,便于用戶可以自定義操作bean(例如BeanPostProcessor)

在spring中確實發(fā)現(xiàn)了很多的繼承自BeanPostProcessor的類版姑,beanfactory通過getBeanPostProcessor方法獲得所有的BeanPostProcessor,然后通過類型篩選取得需要的對象迟郎,依次處理被管理的bean剥险。如我們之前說的Spring 鉤子之BeanFactoryPostProcessor和BeanPostProcessor的源碼學(xué)習(xí) 同樣是BeanPostProcessor的一種∠苄ぃ基本上所有的使用的套路都是這樣的表制,如下圖中還存在自定義的CustomBeanPostProcessor

image.png

PS:在idea中可以通過Ctrl+H查看該類繼承圖

相互引用的情況,A引用B匈庭,B引用A該如何解決

這里存在兩種依賴情況

  • 構(gòu)造器依賴

構(gòu)造器依賴意味著引用的對象必須實例化夫凸,可是這樣就導(dǎo)致了A等著B的實例化,B又等著A的實例化阱持,造成了死鎖一般的情況夭拌,會提示錯誤

  • set依賴

set依賴原則上來說是沒問題的,只需要選擇一個先實例化即可衷咽,但是官方并不推薦這樣做鸽扁,而是推薦修改業(yè)務(wù)邏輯,使用其他方法實現(xiàn)

這一節(jié)基本上就到此結(jié)束了镶骗,大概的說了一些之前忽略的點桶现,其實還有幾個點可以細(xì)說,限于精力鼎姊,后續(xù)會繼續(xù)介紹同名Bean的處理情況以及BeanPostProcessor

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末骡和,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子相寇,更是在濱河造成了極大的恐慌慰于,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件唤衫,死亡現(xiàn)場離奇詭異婆赠,居然都是意外死亡,警方通過查閱死者的電腦和手機佳励,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進(jìn)店門休里,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人赃承,你說我怎么就攤上這事妙黍。” “怎么了瞧剖?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵废境,是天一觀的道長。 經(jīng)常有香客問我,道長噩凹,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任毡咏,我火速辦了婚禮驮宴,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘呕缭。我一直安慰自己堵泽,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布恢总。 她就那樣靜靜地躺著迎罗,像睡著了一般。 火紅的嫁衣襯著肌膚如雪片仿。 梳的紋絲不亂的頭發(fā)上纹安,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天,我揣著相機與錄音砂豌,去河邊找鬼厢岂。 笑死,一個胖子當(dāng)著我的面吹牛阳距,可吹牛的內(nèi)容都是我干的塔粒。 我是一名探鬼主播,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼筐摘,長吁一口氣:“原來是場噩夢啊……” “哼卒茬!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起咖熟,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤圃酵,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后球恤,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體辜昵,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年咽斧,在試婚紗的時候發(fā)現(xiàn)自己被綠了堪置。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡张惹,死狀恐怖舀锨,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情宛逗,我是刑警寧澤坎匿,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響替蔬,放射性物質(zhì)發(fā)生泄漏告私。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一承桥、第九天 我趴在偏房一處隱蔽的房頂上張望驻粟。 院中可真熱鬧,春花似錦凶异、人聲如沸蜀撑。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽酷麦。三九已至,卻和暖如春喉恋,著一層夾襖步出監(jiān)牢的瞬間沃饶,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工瀑晒, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留绍坝,地道東北人。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓苔悦,卻偏偏與公主長得像轩褐,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子玖详,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,916評論 2 344

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理把介,服務(wù)發(fā)現(xiàn),斷路器蟋座,智...
    卡卡羅2017閱讀 134,601評論 18 139
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,748評論 6 342
  • 文章作者:Tyan博客:noahsnail.com | CSDN | 簡書 3.8 Container Exten...
    SnailTyan閱讀 1,228評論 0 6
  • Spring容器高層視圖 Spring 啟動時讀取應(yīng)用程序提供的Bean配置信息拗踢,并在Spring容器中生成一份相...
    Theriseof閱讀 2,796評論 1 24
  • 今天來聊聊大家都很熟悉的一個概念:復(fù)利效應(yīng)。簡單的用公式表示就是:收益=本金x年利率^n向臀。 很清楚的看到這里頭有三...
    程影閱讀 327評論 0 0