欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring IOC基于注解啟動(dòng)示例詳析

 更新時(shí)間:2019年03月03日 11:51:45   作者:神易風(fēng)  
這篇文章主要給大家介紹了Spring IOC基于注解啟動(dòng)的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧

Spring 基于注解啟動(dòng)

主要有兩個(gè)Class實(shí)現(xiàn)注解啟動(dòng)

  • AnnotationConfigApplicationContext
  • AnnotationConfigWebApplicationContext

我們以AnnotationConfigApplicationContext 為研究對(duì)象


AnnotationConfigApplicationContext.png

引入Spring 最小依賴

 <dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-context</artifactId>
  <version>${spring.version}</version>
 </dependency>

編寫器啟動(dòng)代碼

 public static void main(String[] args) {
 AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext();
 applicationContext.register(BeanConfig.class);
 applicationContext.refresh();
 Date date = applicationContext.getBean("date",Date.class);
 System.out.println(date);
 }

AnnotationConfigApplicationContext 構(gòu)造函數(shù)

 public AnnotationConfigApplicationContext() {
    //負(fù)責(zé)注冊(cè)Class ,讀取器
  this.reader = new AnnotatedBeanDefinitionReader(this);
    //負(fù)責(zé)掃描指定類路徑下的Class,注冊(cè)bean
  this.scanner = new ClassPathBeanDefinitionScanner(this);
 }

AnnotatedBeanDefinitionReader 構(gòu)造方法

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

 public AnnotatedBeanDefinitionReader(BeanDefinitionRegistry registry, Environment environment) {
  Assert.notNull(registry, "BeanDefinitionRegistry must not be null");
  Assert.notNull(environment, "Environment must not be null");
  this.registry = registry;
    //初始化ConditionEvaluator
  this.conditionEvaluator = new ConditionEvaluator(registry, environment, null);
     
  /** 在給定的注冊(cè)表中註冊(cè)所有相關(guān)的post processors
   * 判斷容器是否已經(jīng)存在給定注冊(cè)表的bean,如果沒有注冊(cè)bean,并將bean放入容器中
   * 把所有的處理處理器列出來
   * ConfigurationClassPostProcessor 內(nèi)部管理的配置注解處理器
   * AutowiredAnnotationBeanPostProcessor 內(nèi)部管理@Autowired 的處理器
   * RequiredAnnotationBeanPostProcessor @Required的處理器
   * CommonAnnotationBeanPostProcessor JSR-250注解處理器 ,先判斷是否支持jsr,如果支持注冊(cè)
   * PersistenceAnnotationBeanPostProcessor JPA管理 先使用類加載器查找是否存在,如果有這個(gè)包則注冊(cè)
   * EventListenerMethodProcessor @EventListener的處理器
   * DefaultEventListenerFactory 管理EventListenerFactory處理器
   */
  AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry);
 }

ConditionEvaluator 這個(gè)對(duì)象干什么,點(diǎn)擊進(jìn)去

 public ConditionEvaluator(@Nullable BeanDefinitionRegistry registry,
   @Nullable Environment environment, @Nullable ResourceLoader resourceLoader) {

  this.context = new ConditionContextImpl(registry, environment, resourceLoader); 
 }
  //ConditionContextImpl 實(shí)現(xiàn)了ConditionContext接口,ConditionEvaluator靜態(tài)內(nèi)部類
  public ConditionContextImpl(@Nullable BeanDefinitionRegistry registry,
    @Nullable Environment environment, @Nullable ResourceLoader resourceLoader) {

   this.registry = registry;
   this.beanFactory = deduceBeanFactory(registry);
   this.environment = (environment != null ? environment : deduceEnvironment(registry));
   this.resourceLoader = (resourceLoader != null ? resourceLoader : deduceResourceLoader(registry));
   this.classLoader = deduceClassLoader(resourceLoader, this.beanFactory);
  }

可以知道ConditionEvaluator使用外部傳參的方法初始化了Spring容器頂級(jí)對(duì)象
BeanFactory,Environment,ResourceLoader,ClassLoader。在將這些傳給ConditionContextImpl為接下來的解析@Conditional注解做好準(zhǔn)備

ClassPathBeanDefinitionScanner構(gòu)造函數(shù)

 public ClassPathBeanDefinitionScanner(BeanDefinitionRegistry registry) {
  this(registry, true);
 }

 public ClassPathBeanDefinitionScanner(BeanDefinitionRegistry registry, boolean useDefaultFilters) {
  this(registry, useDefaultFilters, getOrCreateEnvironment(registry));
 }

 public ClassPathBeanDefinitionScanner(BeanDefinitionRegistry registry, boolean useDefaultFilters,
   Environment environment) {
  this(registry, useDefaultFilters, environment,
    (registry instanceof ResourceLoader ? (ResourceLoader) registry : null));
 }

 public ClassPathBeanDefinitionScanner(BeanDefinitionRegistry registry, boolean useDefaultFilters,
   Environment environment, @Nullable ResourceLoader resourceLoader) {
  Assert.notNull(registry, "BeanDefinitionRegistry must not be null");
  this.registry = registry;
  if (useDefaultFilters) {
   registerDefaultFilters();
  }
  setEnvironment(environment);
  setResourceLoader(resourceLoader);
 }

 protected void registerDefaultFilters() {
  this.includeFilters.add(new AnnotationTypeFilter(Component.class));
  ClassLoader cl = ClassPathScanningCandidateComponentProvider.class.getClassLoader();
  try {
   this.includeFilters.add(new AnnotationTypeFilter(
     ((Class<? extends Annotation>) ClassUtils.forName("javax.annotation.ManagedBean", cl)), false));
   logger.debug("JSR-250 'javax.annotation.ManagedBean' found and supported for component scanning");
  }
  catch (ClassNotFoundException ex) {
  }
  try {
   this.includeFilters.add(new AnnotationTypeFilter(
     ((Class<? extends Annotation>) ClassUtils.forName("javax.inject.Named", cl)), false));
   logger.debug("JSR-330 'javax.inject.Named' annotation found and supported for component scanning");
  }
  catch (ClassNotFoundException ex) {
   // JSR-330 API not available - simply skip.
  }
 }

繞了地球幾圈了,其實(shí)就是將Spring 頂級(jí)接口 Environment,ResourceLoader賦值,使用默認(rèn)注解過濾器,首先將@Component加入List中,判斷當(dāng)前環(huán)境是否支持JSR-250,JSR-330,相應(yīng)加入過濾器中。也就是這個(gè)掃描器默認(rèn)只掃描@Component或者JSR-250,JSR-330的標(biāo)記的Class。

applicationContext.register(BeanConfig.class)

 public void register(Class<?>... annotatedClasses) {
  Assert.notEmpty(annotatedClasses, "At least one annotated class must be specified");
  this.reader.register(annotatedClasses); //調(diào)用 剛剛初始化讀取器
 }
           |
============================AnnotatedBeanDefinitionReader 讀取器代碼======================================================================================================
  public void register(Class<?>... annotatedClasses) {
  for (Class<?> annotatedClass : annotatedClasses) {
   registerBean(annotatedClass);
  }
  }

 public void registerBean(Class<?> annotatedClass) {
  doRegisterBean(annotatedClass, null, null, null);
 }
  
 /**
 *從給定的bean解析Class給定的注解,執(zhí)行相應(yīng)的初始化,保存到Spring容器中
 */
 <T> void doRegisterBean(Class<T> annotatedClass, @Nullable Supplier<T> instanceSupplier, @Nullable String name,
   @Nullable Class<? extends Annotation>[] qualifiers, BeanDefinitionCustomizer... definitionCustomizers) {

  //根據(jù)Class的Annotated 得出元數(shù)據(jù) AnnotationMetadata
  AnnotatedGenericBeanDefinition abd = new AnnotatedGenericBeanDefinition(annotatedClass);
  /**
  * 判斷注冊(cè)的Class 是否包含@Conditional注解,如果有獲取全部value,放入List中
  * 排序后,遍歷所有的Conditiion的實(shí)現(xiàn),使用反射獲取對(duì)象,執(zhí)行matches方法,
  * 如果發(fā)現(xiàn)有返回false,中斷循環(huán)直接返回true,
  */
  if (this.conditionEvaluator.shouldSkip(abd.getMetadata())) { //如果 @Conditional條件不滿足,不進(jìn)行注冊(cè)
   return;
  }

  abd.setInstanceSupplier(instanceSupplier);
  //解析Class是否有@Scope,解析@Scope注解返回ScopeMetadata對(duì)象,沒有直接返回空
  ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(abd);
  abd.setScope(scopeMetadata.getScopeName());
  //判斷注解上Value是否有值,有就使用這個(gè)作為BeanName,沒有則取類名 
  String beanName = (name != null ? name : this.beanNameGenerator.generateBeanName(abd, this.registry));
  //繼續(xù)解析AnnotationMetadata的@Lazy,@Primary,@DependsOn,@Role,@Description的注解,放入結(jié)果放入對(duì)象的屬性中
  AnnotationConfigUtils.processCommonDefinitionAnnotations(abd);
  //這個(gè)類只是BeanDefinition 包裝類
  BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(abd, beanName);
  //是否需要代理類,如果是則修改內(nèi)部屬性,重新生成BeanDefinition 對(duì)象
  definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
  //調(diào)用DefaultListableBeanFactory.registerBeanDefinition的方法,做一些安全性校驗(yàn)再,將definitionHolder 放入register容器中
  BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, this.registry);
 }

這個(gè)方法就是將注冊(cè)的Bean,解析Class上的注解,初始化注解數(shù)據(jù),做相應(yīng)處理,轉(zhuǎn)化成BeanDefinition ,放入Spring 容器中保存起來。

我們看下BeanDefinition是怎么實(shí)現(xiàn)注冊(cè)到Spring的容器中,主要由DefaultListableBeanFactory.registerBeanDefinition來實(shí)現(xiàn)

public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)
   throws BeanDefinitionStoreException {

  Assert.hasText(beanName, "Bean name must not be empty");
  Assert.notNull(beanDefinition, "BeanDefinition must not be null");

  if (beanDefinition instanceof AbstractBeanDefinition) {
   try {
    //對(duì)beanDefinition 進(jìn)行校驗(yàn)判斷MethodOverrides不能為空,必須擁有工廠方法
    ((AbstractBeanDefinition) beanDefinition).validate();
   }
   catch (BeanDefinitionValidationException ex) {
    throw new BeanDefinitionStoreException(beanDefinition.getResourceDescription(), beanName,
      "Validation of bean definition failed", ex);
   }
  }

  BeanDefinition oldBeanDefinition;

  oldBeanDefinition = this.beanDefinitionMap.get(beanName); 
  if (oldBeanDefinition != null) {
    //這個(gè)方法是判斷是否允許出現(xiàn)重名bean,并且是不同的定義bean,是否可以覆蓋前者
   if (!isAllowBeanDefinitionOverriding()) {
    throw new BeanDefinitionStoreException(beanDefinition.getResourceDescription(), beanName,
      "Cannot register bean definition [" + beanDefinition + "] for bean '" + beanName +
      "': There is already [" + oldBeanDefinition + "] bound.");
   }
   else if (oldBeanDefinition.getRole() < beanDefinition.getRole()) {
    // e.g. was ROLE_APPLICATION, now overriding with ROLE_SUPPORT or ROLE_INFRASTRUCTURE
    if (this.logger.isWarnEnabled()) {
     this.logger.warn("Overriding user-defined bean definition for bean '" + beanName +
       "' with a framework-generated bean definition: replacing [" +
       oldBeanDefinition + "] with [" + beanDefinition + "]");
    }
   }
   else if (!beanDefinition.equals(oldBeanDefinition)) {
    if (this.logger.isInfoEnabled()) {
     this.logger.info("Overriding bean definition for bean '" + beanName +
       "' with a different definition: replacing [" + oldBeanDefinition +
       "] with [" + beanDefinition + "]");
    }
   }
   else {
    if (this.logger.isDebugEnabled()) {
     this.logger.debug("Overriding bean definition for bean '" + beanName +
       "' with an equivalent definition: replacing [" + oldBeanDefinition +
       "] with [" + beanDefinition + "]");
    }
   }
   this.beanDefinitionMap.put(beanName, beanDefinition);
  }
  else {
   //調(diào)用alreadyCreated.isEmpty(),alreadyCreated Set對(duì)象,保存已經(jīng)創(chuàng)建beanName
   //文檔中表示created,跟這里注冊(cè)應(yīng)該不是同一個(gè)行為,這個(gè)要看到后面才知道什么意思
   if (hasBeanCreationStarted()) {
    synchronized (this.beanDefinitionMap) {//更新數(shù)據(jù)
     this.beanDefinitionMap.put(beanName, beanDefinition);
     List<String> updatedDefinitions = new ArrayList<>(this.beanDefinitionNames.size() + 1);
     updatedDefinitions.addAll(this.beanDefinitionNames);
     updatedDefinitions.add(beanName);
     this.beanDefinitionNames = updatedDefinitions;
     if (this.manualSingletonNames.contains(beanName)) {
      Set<String> updatedSingletons = new LinkedHashSet<>(this.manualSingletonNames);
      updatedSingletons.remove(beanName);
      this.manualSingletonNames = updatedSingletons;
     }
    }
   }
   else { 
    //Spring beanDefinition 容器,一個(gè)Map轉(zhuǎn)載
    this.beanDefinitionMap.put(beanName, beanDefinition);
    //保存beanName,主要用于記錄每個(gè)bean注冊(cè)順序
    this.beanDefinitionNames.add(beanName);
    //刪除單例,注冊(cè)成一個(gè)普通bean
    this.manualSingletonNames.remove(beanName);
   }
   this.frozenBeanDefinitionNames = null;
  }

  if (oldBeanDefinition != null || containsSingleton(beanName)) { 
   //更新Spring容器里beanName
   resetBeanDefinition(beanName);
  }
 }

將beanDefinition注冊(cè)到Spring容器中,并沒有太多復(fù)雜的邏輯,只是做一些安全性的檢查。

BeanDefinition

一個(gè)BeanDefinition描述了一個(gè)bean的實(shí)例,包括屬性值,構(gòu)造方法參數(shù)值和繼承自它的類的更多信息。BeanDefinition僅僅是一個(gè)最簡(jiǎn)單的接口,主要功能是允許BeanFactoryPostProcessor 例如PropertyPlaceHolderConfigure 能夠檢索并修改屬性值和別的bean的元數(shù)據(jù)(譯注)

Spring 容器beanDefinition主要分為RootBeanDefinition,AnnotatedGenericBeanDefinition這兩種

  • RootBeanDefinition     Spring Factory中的特定bean
  • AnnotatedGenericBeanDefinition      用戶自定義bean

Spring 啟動(dòng)流程總結(jié)

AnnotationConfigApplicationContext 初始化.png

這些BeanDefinition只是放入到Spirng 容器中,并沒有進(jìn)行任何初始化對(duì)象的操作,真正的IOC操作都在refresh(),這個(gè)方法有空再進(jìn)行分析。

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對(duì)腳本之家的支持。

相關(guān)文章

  • 一次因Java應(yīng)用造成CPU過高的排查實(shí)踐過程

    一次因Java應(yīng)用造成CPU過高的排查實(shí)踐過程

    一個(gè)應(yīng)用占用CPU很高,除了確實(shí)是計(jì)算密集型應(yīng)用之外,通常原因都是出現(xiàn)了死循環(huán)。下面這篇文章主要給大家介紹了一次因Java應(yīng)用造成CPU過高的排查實(shí)踐過程,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2018-11-11
  • 在controller中如何設(shè)置接收參數(shù)的默認(rèn)值

    在controller中如何設(shè)置接收參數(shù)的默認(rèn)值

    這篇文章主要介紹了在controller中如何設(shè)置接收參數(shù)的默認(rèn)值,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • 一文搞懂Java設(shè)計(jì)模式之責(zé)任鏈模式

    一文搞懂Java設(shè)計(jì)模式之責(zé)任鏈模式

    這篇文章主要給大家介紹了關(guān)于Java設(shè)計(jì)模式之責(zé)任鏈模式的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-12-12
  • java 中死鎖問題的實(shí)例詳解

    java 中死鎖問題的實(shí)例詳解

    這篇文章主要介紹了java 中死鎖問題的實(shí)例詳解的相關(guān)資料,希望通過本文大家能夠理解掌握死鎖的問題,需要的朋友可以參考下
    2017-09-09
  • java用靜態(tài)工廠代替構(gòu)造函數(shù)使用方法和優(yōu)缺點(diǎn)

    java用靜態(tài)工廠代替構(gòu)造函數(shù)使用方法和優(yōu)缺點(diǎn)

    這篇文章主要介紹了java用靜態(tài)工廠代替構(gòu)造函數(shù)使用方法和優(yōu)缺點(diǎn),需要的朋友可以參考下
    2014-02-02
  • Java的Hibernate框架中的組合映射學(xué)習(xí)教程

    Java的Hibernate框架中的組合映射學(xué)習(xí)教程

    組合映射即是指主對(duì)象和子對(duì)象關(guān)聯(lián)且擁有相同的生命周期的映射關(guān)系,這里我們將舉一些數(shù)據(jù)操作的實(shí)例,來講解Java的Hibernate框架中的組合映射學(xué)習(xí)教程
    2016-07-07
  • Java自定義一個(gè)變長(zhǎng)數(shù)組的思路與代碼

    Java自定義一個(gè)變長(zhǎng)數(shù)組的思路與代碼

    有時(shí)我們希望將把數(shù)據(jù)保存在單個(gè)連續(xù)的數(shù)組中,以便快速、便捷地訪問數(shù)據(jù),但這需要調(diào)整數(shù)組大小或者對(duì)其擴(kuò)展,下面這篇文章主要給大家介紹了關(guān)于Java自定義一個(gè)變長(zhǎng)數(shù)組的思路與代碼,需要的朋友可以參考下
    2022-12-12
  • springcloud引入spring-cloud-starter-openfeign失敗的解決

    springcloud引入spring-cloud-starter-openfeign失敗的解決

    這篇文章主要介紹了springcloud?引入spring-cloud-starter-openfeign失敗的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • springboot項(xiàng)目打包成jar包的圖文教程

    springboot項(xiàng)目打包成jar包的圖文教程

    有時(shí)候我們會(huì)用IDEA來開發(fā)一些小工具,需要打成可運(yùn)行的JAR包,這篇文章主要給大家介紹了關(guān)于springboot項(xiàng)目打包成jar包的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-06-06
  • java bootclasspath的具體用法

    java bootclasspath的具體用法

    本文主要介紹了java bootclasspath的具體用法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-01-01

最新評(píng)論