成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

揭秘 @Autowired 自動裝配原理與實(shí)現(xiàn)機(jī)制

開發(fā) 前端
在Java開發(fā)中,Spring框架憑借強(qiáng)大的依賴注入(Dependency Injection,DI)功能極大地提升了代碼的可維護(hù)性和可擴(kuò)展性。其中,@Autowired注解作為Spring實(shí)現(xiàn)依賴注入的核心手段之一,被開發(fā)者廣泛使用。

前言

在Java開發(fā)中,Spring框架憑借強(qiáng)大的依賴注入(Dependency Injection,DI)功能極大地提升了代碼的可維護(hù)性和可擴(kuò)展性。其中,@Autowired注解作為Spring實(shí)現(xiàn)依賴注入的核心手段之一,被開發(fā)者廣泛使用。

但@Autowired究竟是如何實(shí)現(xiàn)自動裝配的呢?本文將深入剖析其原理與實(shí)現(xiàn)機(jī)制,揭開它的神秘面紗。

自動裝配的核心流程

啟動容器時的掃描與解析

當(dāng)Spring容器啟動時,會根據(jù)配置(如XML 配置或注解配置)掃描指定包路徑下的類。對于被@Component、@Service、@Repository、@Controller等注解標(biāo)注的類,Spring會將其注冊為Bean,并存儲在容器的BeanDefinition注冊表中。在這個過程中,如果遇到帶有@Autowired注解的字段、方法或構(gòu)造函數(shù),會記錄下依賴關(guān)系信息。

實(shí)例化與依賴注入

在創(chuàng)建Bean實(shí)例時,Spring會根據(jù)BeanDefinition中的信息,首先實(shí)例化Bean(對于單例Bean,只會實(shí)例化一次)。在實(shí)例化完成后,Spring會處理@Autowired注解。對于字段上的@Autowired,Spring會在容器中查找與字段類型匹配的Bean。查找過程中,會優(yōu)先按照類型匹配,如果存在多個同類型的Bean,則會根據(jù)字段名或使用@Qualifier注解指定的名稱進(jìn)一步篩選。找到匹配的Bean后,將其注入到對應(yīng)字段。

循環(huán)依賴處理

圖片圖片

在自動裝配過程中,可能會遇到循環(huán)依賴的情況,即兩個或多個Bean之間相互依賴。Spring容器通過三級緩存機(jī)制來處理循環(huán)依賴問題。一級緩存存儲完全初始化好的Bean;二級緩存存儲早期暴露的Bean(尚未完成屬性填充和初始化);三級緩存存儲Bean 的創(chuàng)建工廠。當(dāng)檢測到循環(huán)依賴時,Spring會從二級緩存中獲取早期暴露的Bean,避免因循環(huán)依賴導(dǎo)致的死循環(huán)問題。

Bean的初始化過程

Spring的可以分為5個小的階段:實(shí)例化、初始化、注冊Destruction回調(diào)、Bean的正常使用以及Bean的銷毀。

圖片圖片

  • 實(shí)例化(Instantiation):
  • 實(shí)例化是創(chuàng)建對象的過程。在Spring中,這通常指的是通過調(diào)用類的構(gòu)造器來創(chuàng)建Bean的實(shí)例。這是對象生命周期的開始階段。對應(yīng)doCreateBean中的createBeanInstance方法。
  • 初始化(Initialization):
  • 初始化是在Bean實(shí)例創(chuàng)建后,進(jìn)行一些設(shè)置或準(zhǔn)備工作的過程。在Spring中,包括設(shè)置Bean的屬性,調(diào)用各種前置&后置處理器。對應(yīng)doCreateBean中的populateBean和initializeBean方法。

自動裝配其實(shí)發(fā)生在Instantiation -> Populate -> Initialization的過程中,屬于Bean的生命周期中的populate階段。

  • 在實(shí)例化Bean的時候在createBeanInstance方法中會調(diào)用 AutowiredAnnotationBeanPostProcessor中的方法來獲取需要自動注入的構(gòu)造方法,AutowiredAnnotationBeanPostProcessor的方法來完成對所有需要自動注入的屬性和方法的解析和緩存。
  • populatedBean方法中調(diào)用到AutowiredAnnotationBeanPostProcessor中的方法來完成需要自動注入屬性的注入工作。
protected void populateBean(String beanName, RootBeanDefinition mbd, BeanWrapper bw) {
    // 獲取Bean定義中的屬性值    
    PropertyValues pvs = mbd.getPropertyValues();
    // 如果BeanWrapper為空,則無法設(shè)置屬性值    
    if (bw == null) {
        if (!pvs.isEmpty()) {
            throw new BeanCreationException(
                    mbd.getResourceDescription(), beanName, "Cannot apply property values to null instance");
        } else {
            // 對于null實(shí)例,跳過設(shè)置屬性階段            
            return;
        }
    }
    // 在設(shè)置屬性之前,給InstantiationAwareBeanPostProcessors機(jī)會修改Bean狀態(tài)    
    // 這可以用于支持字段注入等樣式    
    boolean continueWithPropertyPopulation = true;
    // 如果Bean不是合成的,并且存在InstantiationAwareBeanPostProcessor,執(zhí)行后續(xù)處理    
    if (!mbd.isSynthetic() && hasInstantiationAwareBeanPostProcessors()) {
        for (BeanPostProcessor bp : getBeanPostProcessors()) {
            if (bp instanceof InstantiationAwareBeanPostProcessor) {
                InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp;
                if (!ibp.postProcessAfterInstantiation(bw.getWrappedInstance(), beanName)) {
                    continueWithPropertyPopulation = false;
                    break;
                }
            }
        }
    }
    // 如果上述處理后決定不繼續(xù),則返回    
    if (!continueWithPropertyPopulation) {
        return;
    }
    // 根據(jù)自動裝配模式(按名稱或類型),設(shè)置相關(guān)的屬性值    
    if (mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_NAME ||
            mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_TYPE) {
        MutablePropertyValues newPvs = new MutablePropertyValues(pvs);
        // 如果是按名稱自動裝配,添加相應(yīng)的屬性值        
        if (mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_NAME) {
            autowireByName(beanName, mbd, bw, newPvs);
        }
        // 如果是按類型自動裝配,添加相應(yīng)的屬性值        
        if (mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_TYPE) {
            autowireByType(beanName, mbd, bw, newPvs);
        }
        pvs = newPvs;
    }
    // 檢查是否需要進(jìn)行依賴性檢查    
    boolean hasInstAwareBpps = hasInstantiationAwareBeanPostProcessors();
    boolean needsDepCheck = (mbd.getDependencyCheck() != RootBeanDefinition.DEPENDENCY_CHECK_NONE);
    // 如果需要,則進(jìn)行依賴性檢查    
    if (hasInstAwareBpps || needsDepCheck) {
        PropertyDescriptor[] filteredPds = filterPropertyDescriptorsForDependencyCheck(bw, mbd.allowCaching);
        if (hasInstAwareBpps) {
            for (BeanPostProcessor bp : getBeanPostProcessors()) {
                if (bp instanceof InstantiationAwareBeanPostProcessor) {
                    InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp;
                    pvs = ibp.postProcessPropertyValues(pvs, filteredPds, bw.getWrappedInstance(), beanName);
                    if (pvs == null) {
                        return;
                    }
                }
            }
        }
        if (needsDepCheck) {
            checkDependencies(beanName, mbd, filteredPds, pvs);
        }
    }
    // 應(yīng)用屬性值    
    applyPropertyValues(beanName, mbd, bw, pvs);
}

@Autowired 實(shí)現(xiàn)的關(guān)鍵源碼解析

@Autowired注解的實(shí)現(xiàn)主要涉及到AutowiredAnnotationBeanPostProcessor類,它繼承自InstantiationAwareBeanPostProcessorAdapter(后期被棄用,改實(shí)現(xiàn)SmartInstantiationAwareBeanPostProcessor),該類實(shí)現(xiàn)了postProcessProperties方法,用于處理Bean的屬性注入。

@Override
public PropertyValues postProcessProperties(PropertyValues pvs, Object bean, String beanName) {
    InjectionMetadata metadata = findAutowiringMetadata(beanName, bean.getClass(), pvs);
    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;
}

在postProcessProperties方法中,首先通過findAutowiringMetadata方法獲取與@Autowired相關(guān)的元數(shù)據(jù)(包括要注入的字段、方法等信息),然后調(diào)用metadata.inject方法進(jìn)行實(shí)際的依賴注入操作。metadata.inject方法會遍歷需要注入的元素,通過反射機(jī)制獲取對應(yīng)的Bean并完成注入。

總結(jié)

如果你在面試中遇到這個問題,可以這樣回答:

  1. @Autowired是Spring提供的自動裝配注解,它通過AutowiredAnnotationBeanPostProcessor實(shí)現(xiàn)依賴注入。
  2. 這個處理器在Bean初始化階段介入,掃描字段或方法的注解,找到需要注入的地方,然后根據(jù)類型從容器中查找Bean,通過反射完成注入。
  3. 如果存在多個實(shí)現(xiàn),可以通過@Qualifier或@Primary來指定注入對象。構(gòu)造器注入則在實(shí)例化時完成,推薦用于強(qiáng)依賴場景。
責(zé)任編輯:武曉燕 來源: 一安未來
相關(guān)推薦

2025-01-14 10:09:43

硬中斷Linux系統(tǒng)

2022-08-08 07:33:11

自動裝配Java容器

2024-12-25 16:01:01

2020-09-22 08:52:50

平臺實(shí)現(xiàn)消息

2012-11-28 10:37:25

Linux集群安裝配置

2011-06-13 10:21:25

QT 信號 槽機(jī)制

2024-10-10 17:46:06

2021-07-19 09:25:19

數(shù)據(jù)庫MySQL技術(shù)

2022-04-08 08:32:40

mobx狀態(tài)管理庫redux

2025-05-27 01:00:00

2010-04-29 10:11:07

Linux集群

2010-08-06 09:45:50

Flex事件機(jī)制

2012-02-17 10:50:10

Java

2010-08-06 10:15:35

Flex綁定

2024-10-10 12:12:45

SpringAI版本

2021-05-13 23:30:17

JavaScript 原理揭秘

2020-02-25 16:48:35

AndroidGoogle 移動系統(tǒng)

2009-09-04 10:37:50

Java堆棧溢出

2015-03-27 09:50:37

2021-12-20 00:03:38

Webpack運(yùn)行機(jī)制
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 无码国模国产在线观看 | 色综合一区二区 | 国产精品我不卡 | 欧美一级欧美三级在线观看 | 亚洲色图第一页 | 国产精品视频久久 | 久久成人免费视频 | 国产美女福利在线观看 | 天堂一区 | 久久久久久国产 | 免费成人在线网站 | 欧美视频三区 | 成人福利片 | 国产一区在线视频 | 综合一区二区三区 | 91精品国产综合久久久久久蜜臀 | 日韩电影中文字幕 | 草草视频在线观看 | 国产美女黄色片 | www.xxxx欧美 | 波霸ol一区二区 | 国产精品视频一 | 亚洲国产精品久久久久婷婷老年 | 在线观看国产 | 国产精品综合 | 免费欧美 | 中文字幕高清在线 | 国产精品99 | 在线精品亚洲欧美日韩国产 | 国产精品美女久久久久久久网站 | 国产免费黄网 | 精品国产一区二区国模嫣然 | 免费一区二区 | 成人免费观看网站 | 麻豆视频在线免费观看 | 国产精品高潮呻吟久久久久 | 国产日韩欧美二区 | 国产日韩欧美精品 | 日本精品一区二区三区在线观看 | 99re热精品视频 | 国产影音先锋 |