91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Spring源碼解析容器初始化構造的方法是什么

發布時間:2022-07-07 13:58:55 來源:億速云 閱讀:128 作者:iii 欄目:開發技術

這篇“Spring源碼解析容器初始化構造的方法是什么”文章的知識點大部分人都不太理解,所以小編給大家總結了以下內容,內容詳細,步驟清晰,具有一定的借鑒價值,希望大家閱讀完這篇文章能有所收獲,下面我們一起來看看這篇“Spring源碼解析容器初始化構造的方法是什么”文章吧。

在開始進行源碼學習前,首先再回顧一下三種Spring編程風格:

  • 基于Schema,即通過xml標簽的配置方式

  • 基于Annotation的注解技術,使用@Component等注解配置bean

  • 基于Java Config,簡單來說就是使用@Configuration@Bean進行配置

基于注解的方式需要通過xml或java config來開啟。

在使用xml時,需要手動開啟對注解的支持:

<context: annotation-config/>

當然,如果在xml中配置了掃描包,現在也可以光添加下面這一行,這行代碼中已經包含了注解的開啟功能。

<context: component-sacn base-package="com"/>

如果你使用的是下面AnnotationConfigApplicationContext這種方式,那么就不需要添加任何操作了,其中已經包含了對注解的支持。

AnnotationConfigApplicationContext ctx
	=new AnnotationConfigApplicationContext(SpringConfig.class);

在實際使用過程中,三種方式是可以混合使用的,不存在沖突。按照下面這種方式作為AnnotationConfigApplicationContext傳入的配置文件,即可實現三種風格的統一使用:

@Configuration
@ComponentScan("com")
@ImportResource("classpath:spring.xml") 
public class SpringConfig{
}

之前也有小伙伴對我說,在開始學習Spring的時候,差點因為配置繁雜的xml被勸退,我也翻閱了一下網上spring入門的技術文章,確實很多還是停留在使用xml的方式上。但是其實如果你翻閱一下spring5的官方文檔,可以看出官方是推薦我們使用注解的方式的。

尤其是現在的Spring Boot更多的是基于注解,省略了很多配置的過程,對新手更加友好,降低了勸退率,所以本文將基于注解的方式進行源碼解析,另外再說明一下本文基于spring-framework-5.0.x源碼。

使用注解的方式初始化一個Spring環境,只需要下面一行代碼:

AnnotationConfigApplicationContext context
    = new AnnotationConfigApplicationContext(SpringConfig.class);

如果看一下它的構造方法,那么可以將它做的工作拆分為三步,為了便于理解可以寫成下面的形式,并分為三大模塊分別進行說明。

構造方法

首先看一下AnnotationConfigApplicationContext的繼承關系:

Spring源碼解析容器初始化構造的方法是什么

AnnotationConfigApplicationContext繼承了GenericApplicationContext,那么我們先看GenericApplicationContext的構造方法:

public GenericApplicationContext() {
  this.beanFactory = new DefaultListableBeanFactory();
}

在這里初始化了一個beanFactory的實現類DefaultListableBeanFactory,這就是我們常提到的spring中重要的bean工廠,這里面存放了很多非常重要的數據結構。這里先列出比較重要的beanDefinitionMap,會在后面頻繁使用:

private final Map<String, BeanDefinition> beanDefinitionMap = new ConcurrentHashMap<>(256);
private volatile List<String> beanDefinitionNames = new ArrayList<>(256);

在上面的這個beanDefinitionMap中就維護了beanNameBeanDefinition的對應關系,beanDefinitionNames則是一個存放beanName的List。

AnnotationConfigApplicationContext的構造方法開始分析:

public AnnotationConfigApplicationContext() {
  this.reader = new AnnotatedBeanDefinitionReader(this);
  this.scanner = new ClassPathBeanDefinitionScanner(this);
}

首先實例化了一個AnnotatedBeanDefinitionReader對象,看一下AnnotatedBeanDefinitionReader的構造函數:

public AnnotatedBeanDefinitionReader(BeanDefinitionRegistry registry) {
  this(registry, getOrCreateEnvironment(registry));
}

那么,為什么在這能夠將AnnotationConfigApplicationContext對象作為BeanDefinitionRegistry傳入呢?

回頭看一下繼承關系那張圖,AnnotationConfigApplicationContext繼承了BeanDefinitionRegistry,并且最終實現了接口BeanFactoryBeanFactory可以說是Spring中的頂層類,它是一個工廠,能夠產生bean對象,提供了一個非常重要的方法getBean,會在后面講到。

到這,我們可以得出一個結論:

BeanDefinitionRegistry可以等同于AnnotationConfigApplicationContext ,看做spring的上下文環境。

AnnotatedBeanDefinitionReader在實例化時,會調用registerAnnotationConfigProcessors方法。先看前半段代碼:

public static Set<BeanDefinitionHolder> registerAnnotationConfigProcessors(
    BeanDefinitionRegistry registry, @Nullable Object source) {
    DefaultListableBeanFactory beanFactory = unwrapDefaultListableBeanFactory(registry);
    if (beanFactory != null) {
      if (!(beanFactory.getDependencyComparator() instanceof AnnotationAwareOrderComparator)) {
        beanFactory.setDependencyComparator(AnnotationAwareOrderComparator.INSTANCE);
      }
      if (!(beanFactory.getAutowireCandidateResolver() instanceof ContextAnnotationAutowireCandidateResolver)) {
        beanFactory.setAutowireCandidateResolver(new ContextAnnotationAutowireCandidateResolver());
      }
}

在這里先獲取在父類構造函數中實例好的beanFactory,并為它填充一些屬性:

  • AnnotationAwareOrderComparator:主要用于排序,解析@order@Priority注解

  • ContextAnnotationAutowireCandidateResolver:提供處理延遲加載的功能

再看后半段代碼,下面生成了6個重要類的BeanDefinitionHolder,并存放到一個Set中:

 Set<BeanDefinitionHolder> beanDefs = new LinkedHashSet<>(8);
    if (!registry.containsBeanDefinition(CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME)) {
      RootBeanDefinition def = new RootBeanDefinition(ConfigurationClassPostProcessor.class);
      def.setSource(source);
      beanDefs.add(registerPostProcessor(registry, def, CONFIGURATION_ANNOTATION_PROCESSOR_BEAN_NAME));
    }

    if (!registry.containsBeanDefinition(AUTOWIRED_ANNOTATION_PROCESSOR_BEAN_NAME)) {
      RootBeanDefinition def = new RootBeanDefinition(AutowiredAnnotationBeanPostProcessor.class);
      def.setSource(source);
      beanDefs.add(registerPostProcessor(registry, def, AUTOWIRED_ANNOTATION_PROCESSOR_BEAN_NAME));
    }

    if (!registry.containsBeanDefinition(REQUIRED_ANNOTATION_PROCESSOR_BEAN_NAME)) {
      RootBeanDefinition def = new RootBeanDefinition(RequiredAnnotationBeanPostProcessor.class);
      def.setSource(source);
      beanDefs.add(registerPostProcessor(registry, def, REQUIRED_ANNOTATION_PROCESSOR_BEAN_NAME));
    }

    // Check for JSR-250 support, and if present add the CommonAnnotationBeanPostProcessor.
    if (jsr250Present && !registry.containsBeanDefinition(COMMON_ANNOTATION_PROCESSOR_BEAN_NAME)) {
      RootBeanDefinition def = new RootBeanDefinition(CommonAnnotationBeanPostProcessor.class);
      def.setSource(source);
      beanDefs.add(registerPostProcessor(registry, def, COMMON_ANNOTATION_PROCESSOR_BEAN_NAME));
    }

    // Check for JPA support, and if present add the PersistenceAnnotationBeanPostProcessor.
    if (jpaPresent && !registry.containsBeanDefinition(PERSISTENCE_ANNOTATION_PROCESSOR_BEAN_NAME)) {
      RootBeanDefinition def = new RootBeanDefinition();
      try {
        def.setBeanClass(ClassUtils.forName(PERSISTENCE_ANNOTATION_PROCESSOR_CLASS_NAME,
            AnnotationConfigUtils.class.getClassLoader()));
      }
      catch (ClassNotFoundException ex) {
        throw new IllegalStateException(
            "Cannot load optional framework class: " + PERSISTENCE_ANNOTATION_PROCESSOR_CLASS_NAME, ex);
      }
      def.setSource(source);
      beanDefs.add(registerPostProcessor(registry, def, PERSISTENCE_ANNOTATION_PROCESSOR_BEAN_NAME));
    }

    if (!registry.containsBeanDefinition(EVENT_LISTENER_PROCESSOR_BEAN_NAME)) {
      RootBeanDefinition def = new RootBeanDefinition(EventListenerMethodProcessor.class);
      def.setSource(source);
      beanDefs.add(registerPostProcessor(registry, def, EVENT_LISTENER_PROCESSOR_BEAN_NAME));
    }

    if (!registry.containsBeanDefinition(EVENT_LISTENER_FACTORY_BEAN_NAME)) {
      RootBeanDefinition def = new RootBeanDefinition(DefaultEventListenerFactory.class);
      def.setSource(source);
      beanDefs.add(registerPostProcessor(registry, def, EVENT_LISTENER_FACTORY_BEAN_NAME));
    }

    return beanDefs;
  }

這里是使用RootBeanDefinition來將普通類轉換為BeanDefinition,并進一步封裝成BeanDefinitionHolder。封裝成BeanDefinitionHolder的操作在registerPostProcessor方法中:

 private static BeanDefinitionHolder registerPostProcessor(
      BeanDefinitionRegistry registry, RootBeanDefinition definition, String beanName) {
    definition.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
    registry.registerBeanDefinition(beanName, definition);
    return new BeanDefinitionHolder(definition, beanName);
  }

通過registerBeanDefinition方法將BeanDefinition注冊到spring環境中,這個操作其實就是執行了上面的beanDefinitionMapput操作:

this.beanDefinitionMap.put(beanName, beanDefinition);

在上面的操作全部完成后,在還沒有實例化用戶自定義的bean前,已經有了6個spring自己定義的beanDefinition

用于實現spring自身的初始化:

Spring源碼解析容器初始化構造的方法是什么

這里有必要對BeanDefinition進行一下說明,它是對具有屬性值的bean實例的一個說明,或者說是定義。就像是在java類加載的過程,普通java文件要先生成字節碼文件,再加載到jvm中生成class對象,spring初始化過程中首先要將普通類轉化為BeanDefinition,然后再實例化為bean。

在實例化AnnotatedBeanDefinitionReader完成后,實例化了一個ClassPathBeanDefinitionScanner,可以用來掃描包或者類,并將掃描到的類轉化為BeanDefinition。但是翻閱源碼,我們可以看到實際上掃描包的工作不是這個scanner對象來完成的,而是在后面spring自己實例化了一個ClassPathBeanDefinitionScanner來負責的。

這里的scanner僅僅是對外提供一個擴展,可以讓我們能夠在外部調用AnnotationConfigApplicationContext對象的scan方法,實現包的掃描,

例如:

context.scan("com.hydra");

以上就是關于“Spring源碼解析容器初始化構造的方法是什么”這篇文章的內容,相信大家都有了一定的了解,希望小編分享的內容對大家有幫助,若想了解更多相關的知識內容,請關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

花莲县| 两当县| 天津市| 益阳市| 灌阳县| 瓮安县| 信丰县| 岳阳县| 北京市| 永平县| 龙江县| 沁阳市| 乌什县| 高州市| 旌德县| 宜川县| 太原市| 舞钢市| 和政县| 宣化县| 绥中县| 扶余县| 武鸣县| 东源县| 安溪县| 东明县| 舒城县| 临海市| 岢岚县| 玛曲县| 松阳县| 肃宁县| 康乐县| 霍山县| 景东| 招远市| 普陀区| 沾益县| 阿拉善左旗| 都江堰市| 嵊州市|