1. <ul id="0c1fb"></ul>

      <noscript id="0c1fb"><video id="0c1fb"></video></noscript>
      <noscript id="0c1fb"><listing id="0c1fb"><thead id="0c1fb"></thead></listing></noscript>

      99热在线精品一区二区三区_国产伦精品一区二区三区女破破_亚洲一区二区三区无码_精品国产欧美日韩另类一区

      RELATEED CONSULTING
      相關(guān)咨詢(xún)
      選擇下列產(chǎn)品馬上在線溝通
      服務(wù)時(shí)間:8:30-17:00
      你可能遇到了下面的問(wèn)題
      關(guān)閉右側(cè)工具欄

      新聞中心

      這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷(xiāo)解決方案
      SpringIoc中怎么加載Bean

      這篇文章將為大家詳細(xì)講解有關(guān)Spring Ioc中怎么加載Bean,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個(gè)參考,希望大家閱讀完這篇文章后對(duì)相關(guān)知識(shí)有一定的了解。

      鄂托克網(wǎng)站建設(shè)公司成都創(chuàng)新互聯(lián),鄂托克網(wǎng)站設(shè)計(jì)制作,有大型網(wǎng)站制作公司豐富經(jīng)驗(yàn)。已為鄂托克上千多家提供企業(yè)網(wǎng)站建設(shè)服務(wù)。企業(yè)網(wǎng)站搭建\外貿(mào)網(wǎng)站制作要多少錢(qián),請(qǐng)找那個(gè)售后服務(wù)好的鄂托克做網(wǎng)站的公司定做!

      //真正實(shí)現(xiàn)向IOC容器獲取Bean的功能,也是觸發(fā)依賴(lài)注入功能的地方
      	protected  T doGetBean(final String name, @Nullable final Class requiredType,
      			@Nullable final Object[] args, boolean typeCheckOnly) throws BeansException {
      
      		//根據(jù)指定的名稱(chēng)獲取被管理Bean的名稱(chēng),剝離指定名稱(chēng)中對(duì)容器的相關(guān)依賴(lài)
      		// 如果指定的是別名,將別名轉(zhuǎn)換為規(guī)范的Bean名稱(chēng)
      <1>		final String beanName = transformedBeanName(name);
      		Object bean;
      
      		// Eagerly check singleton cache for manually registered singletons.
      		// 從緩存中獲取已被創(chuàng)建過(guò)的單例Bean
      <2>		Object sharedInstance = getSingleton(beanName);
      		//如果緩存中有
      		if (sharedInstance != null && args == null) {
      			if (logger.isDebugEnabled()) {
      				if (isSingletonCurrentlyInCreation(beanName)) {
      					logger.debug("Returning eagerly cached instance of singleton bean '" + beanName +
      							"' that is not fully initialized yet - a consequence of a circular reference");
      				}
      				else {
      					logger.debug("Returning cached instance of singleton bean '" + beanName + "'");
      				}
      			}
      
      			//注意:BeanFactory是管理容器中Bean的工廠
      			//     FactoryBean是創(chuàng)建創(chuàng)建對(duì)象的工廠Bean,兩者之間有區(qū)別
      
      			//獲取給定Bean的實(shí)例對(duì)象,該對(duì)象要么是 bean 實(shí)例本身,要么就是 FactoryBean 創(chuàng)建的 Bean 對(duì)象
      			//(為什么要再次獲取呢,因?yàn)樯厦娅@取的sharedInstance不一定是完整的)
      <3>			bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);
      		}
      
      		else {
      			// Fail if we're already creating this bean instance:
      			// We're assumably within a circular reference.
      			// 因?yàn)?nbsp;Spring 只解決單例模式下的循環(huán)依賴(lài),在原型模式下如果存在循環(huán)依賴(lài)則會(huì)拋出異常。
      <4>			if (isPrototypeCurrentlyInCreation(beanName)) {
      				throw new BeanCurrentlyInCreationException(beanName);
      			}
      
      			// Check if bean definition exists in this factory.
      			//對(duì)IOC容器中是否存在指定名稱(chēng)的BeanDefinition進(jìn)行檢查,首先檢查是否
      			//能在當(dāng)前的BeanFactory中獲取的所需要的Bean,如果不能則委托當(dāng)前容器
      			//的父級(jí)容器去查找,如果還是找不到則沿著容器的繼承體系向父級(jí)容器查找
      			BeanFactory parentBeanFactory = getParentBeanFactory();
      			//當(dāng)前容器的父級(jí)容器存在,且當(dāng)前容器中不存在指定名稱(chēng)的Bean
      			if (parentBeanFactory != null && !containsBeanDefinition(beanName)) {
      				// Not found -> check parent.
      				//解析指定Bean名稱(chēng)的原始名稱(chēng)
      				String nameToLookup = originalBeanName(name);
      				// 若為 AbstractBeanFactory 類(lèi)型,委托父類(lèi)處理
      				if (parentBeanFactory instanceof AbstractBeanFactory) {
      					return ((AbstractBeanFactory) parentBeanFactory).doGetBean(
      							nameToLookup, requiredType, args, typeCheckOnly);
      				}
      				else if (args != null) {
      					// Delegation to parent with explicit args.
      					//委派父級(jí)容器根據(jù)指定名稱(chēng)和顯式的參數(shù)查找
      					return (T) parentBeanFactory.getBean(nameToLookup, args);
      				}
      				else {
      					// No args -> delegate to standard getBean method.
      					//委派父級(jí)容器根據(jù)指定名稱(chēng)和類(lèi)型查找
      					return parentBeanFactory.getBean(nameToLookup, requiredType);
      				}
      			}
      
      			// 創(chuàng)建的Bean是否需要進(jìn)行類(lèi)型驗(yàn)證,一般不需要
      <5>			if (!typeCheckOnly) {
      				//向容器標(biāo)記指定的Bean已經(jīng)被創(chuàng)建
      				markBeanAsCreated(beanName);
      			}
      
      			try {
      				//從容器中獲取 beanName 相應(yīng)的 GenericBeanDefinition 對(duì)象,并將其轉(zhuǎn)換為 RootBeanDefinition 對(duì)象
      				// 主要解決Bean繼承時(shí)子類(lèi)合并父類(lèi)公共屬性問(wèn)題
      <6>				final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
      				// 檢查給定的合并的 BeanDefinition (是否為抽象類(lèi))
      				checkMergedBeanDefinition(mbd, beanName, args);
      
      				// Guarantee initialization of beans that the current bean depends on.
      				// 處理所依賴(lài)的 bean @DependsOn()
      				// 獲取當(dāng)前Bean所有依賴(lài)Bean的名稱(chēng)
      <7>				String[] dependsOn = mbd.getDependsOn();
      				//如果有依賴(lài)
      				if (dependsOn != null) {
      					for (String dep : dependsOn) {
      						//校驗(yàn)該依賴(lài)是否已經(jīng)注冊(cè)過(guò)給當(dāng)前 Bean
      						if (isDependent(beanName, dep)) {
      							//已注冊(cè),拋出異常
      							throw new BeanCreationException(mbd.getResourceDescription(), beanName,
      									"Circular depends-on relationship between '" + beanName + "' and '" + dep + "'");
      						}
      						//沒(méi)有,則先注冊(cè)依賴(lài)的bean
      						registerDependentBean(dep, beanName);
      						//遞歸調(diào)用getBean(),先生成依賴(lài)的bean
      						getBean(dep);
      					}
      				}
      
      				// Create bean instance.
      				//創(chuàng)建單例Bean
      <8>				if (mbd.isSingleton()) {
      					//這里使用了一個(gè)匿名內(nèi)部類(lèi),創(chuàng)建Bean實(shí)例對(duì)象,并且注冊(cè)給所依賴(lài)的對(duì)象
      					sharedInstance = getSingleton(beanName, () -> {
      						try {
      							//創(chuàng)建一個(gè)指定Bean實(shí)例對(duì)象,如果有父級(jí)繼承,則合并子類(lèi)和父類(lèi)的定義
      							return createBean(beanName, mbd, args);
      						}
      						catch (BeansException ex) {
      							// Explicitly remove instance from singleton cache: It might have been put there
      							// eagerly by the creation process, to allow for circular reference resolution.
      							// Also remove any beans that received a temporary reference to the bean.
      							//顯式地從容器單例模式Bean緩存中清除實(shí)例對(duì)象
      							destroySingleton(beanName);
      							throw ex;
      						}
      					});
      					//獲取給定Bean的實(shí)例對(duì)象
      					bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
      				}
      
      				//創(chuàng)建多例Bean
      				else if (mbd.isPrototype()) {
      					//原型模式(Prototype)是每次都會(huì)創(chuàng)建一個(gè)新的對(duì)象
      					Object prototypeInstance = null;
      					try {
      						//加載前置處理,默認(rèn)的功能是注冊(cè)當(dāng)前創(chuàng)建的原型對(duì)象
      						beforePrototypeCreation(beanName);
      						//創(chuàng)建指定Bean對(duì)象實(shí)例
      						prototypeInstance = createBean(beanName, mbd, args);
      					}
      					finally {
      						//加載后置處理,默認(rèn)的功能告訴IOC容器指定Bean的原型對(duì)象不再創(chuàng)建
      						afterPrototypeCreation(beanName);
      					}
      					//獲取給定Bean的實(shí)例對(duì)象
      					bean = getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
      				}
      
      				//要?jiǎng)?chuàng)建的Bean既不是Singleton也不是Prototype
      				//如:request、session、application等生命周期
      				else {
      					String scopeName = mbd.getScope();
      					final Scope scope = this.scopes.get(scopeName);
      					//Bean定義資源中沒(méi)有配置生命周期范圍,則Bean定義不合法
      					if (scope == null) {
      						throw new IllegalStateException("No Scope registered for scope name '" + scopeName + "'");
      					}
      					try {
      						//這里又使用了一個(gè)匿名內(nèi)部類(lèi),獲取一個(gè)指定生命周期范圍的實(shí)例
      						Object scopedInstance = scope.get(beanName, () -> {
      							//前置處理
      							beforePrototypeCreation(beanName);
      							try {
      								return createBean(beanName, mbd, args);
      							}
      							finally {
      								//后置處理
      								afterPrototypeCreation(beanName);
      							}
      						});
      						//獲取給定Bean的實(shí)例對(duì)象
      						bean = getObjectForBeanInstance(scopedInstance, name, beanName, mbd);
      					}
      					catch (IllegalStateException ex) {
      						throw new BeanCreationException(beanName,
      								"Scope '" + scopeName + "' is not active for the current thread; consider " +
      								"defining a scoped proxy for this bean if you intend to refer to it from a singleton",
      								ex);
      					}
      				}
      			}
      			catch (BeansException ex) {
      				cleanupAfterBeanCreationFailure(beanName);
      				throw ex;
      			}
      		}
      
      		// Check if required type matches the type of the actual bean instance.
      		//對(duì)創(chuàng)建的Bean實(shí)例對(duì)象進(jìn)行類(lèi)型檢查
      <9>		if (requiredType != null && !requiredType.isInstance(bean)) {
      			try {
      				T convertedBean = getTypeConverter().convertIfNecessary(bean, requiredType);
      				if (convertedBean == null) {
      					throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
      				}
      				return convertedBean;
      			}
      			catch (TypeMismatchException ex) {
      				if (logger.isDebugEnabled()) {
      					logger.debug("Failed to convert bean '" + name + "' to required type '" +
      							ClassUtils.getQualifiedName(requiredType) + "'", ex);
      				}
      				throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
      			}
      		}
      		return (T) bean;
      	}

      代碼很長(zhǎng),需要一些耐心,下面我們來(lái)逐步分析這段代碼:

      • <1>處:具體分析,見(jiàn)2.1獲取原始beanName

      • <2>處: 具體分析,見(jiàn)2.2從緩存中獲取單例bean

      • <3>處: 具體分析,見(jiàn)2.3獲取最終的bean實(shí)例對(duì)象

      • <4>處: 具體分析,見(jiàn)2.4原型模式依賴(lài)檢查(Prototype)和從 parentBeanFactory 獲取 Bean

      • <5>處: 具體分析,見(jiàn)2.5標(biāo)記bean為已創(chuàng)建或即將創(chuàng)建

      • <6>處: 具體分析,見(jiàn)2.6獲取BeanDefinition

      • <7>處: 具體分析,見(jiàn)2.7bean依賴(lài)處理

      • <8>處: 具體分析,見(jiàn)2.8不同作用域bean的實(shí)例化

      • <9>處: 具體分析,見(jiàn)2.9類(lèi)型轉(zhuǎn)換

      2.4、原型模式依賴(lài)檢查(Prototype)和從 parentBeanFactory 獲取 Bean

      原型模式依賴(lài)檢查,對(duì)應(yīng)代碼如下:

      if (isPrototypeCurrentlyInCreation(beanName)) {
      	throw new BeanCurrentlyInCreationException(beanName);
      	}

      跟蹤進(jìn)去:

              /** Names of beans that are currently in creation */
      	private final ThreadLocal prototypesCurrentlyInCreation =
      			new NamedThreadLocal<>("Prototype beans currently in creation");
      
      protected boolean isPrototypeCurrentlyInCreation(String beanName) {
      		//從ThreadLocal中取出正在創(chuàng)建的prototype
      		Object curVal = this.prototypesCurrentlyInCreation.get();
      		return (curVal != null &&
      				(curVal.equals(beanName) || (curVal instanceof Set && ((Set) curVal).contains(beanName))));
      	}

      Spring 只處理單例模式下得循環(huán)依賴(lài),對(duì)于原型模式的循環(huán)依賴(lài)直接拋出異常。
      Spring會(huì)把正在創(chuàng)建的原型模式Bean存入ThreadLoacl,在這里通過(guò)ThreadLoacl來(lái)判斷當(dāng)前Bean是否已經(jīng)創(chuàng)建。

      從 parentBeanFactory 獲取 Bean,對(duì)應(yīng)代碼如下:

      // Check if bean definition exists in this factory.
      	//對(duì)IOC容器中是否存在指定名稱(chēng)的BeanDefinition進(jìn)行檢查,首先檢查是否
      	//能在當(dāng)前的BeanFactory中獲取的所需要的Bean,如果不能則委托當(dāng)前容器
      	//的父級(jí)容器去查找,如果還是找不到則沿著容器的繼承體系向父級(jí)容器查找
      	BeanFactory parentBeanFactory = getParentBeanFactory();
      	//當(dāng)前容器的父級(jí)容器存在,且當(dāng)前容器中不存在指定名稱(chēng)的Bean
      	if (parentBeanFactory != null && !containsBeanDefinition(beanName)) {
      		// Not found -> check parent.
      		//解析指定Bean名稱(chēng)的原始名稱(chēng)
      		String nameToLookup = originalBeanName(name);
      		// 若為 AbstractBeanFactory 類(lèi)型,委托父類(lèi)處理
      		if (parentBeanFactory instanceof AbstractBeanFactory) {
      			return ((AbstractBeanFactory) parentBeanFactory).doGetBean(
      					nameToLookup, requiredType, args, typeCheckOnly);
      		}
      		else if (args != null) {
      			// Delegation to parent with explicit args.
      			//委派父級(jí)容器根據(jù)指定名稱(chēng)和顯式的參數(shù)查找
      			return (T) parentBeanFactory.getBean(nameToLookup, args);
      		}
      		else {
      			// No args -> delegate to standard getBean method.
      			//委派父級(jí)容器根據(jù)指定名稱(chēng)和類(lèi)型查找
      			return parentBeanFactory.getBean(nameToLookup, requiredType);
      		}
      	}

      如果當(dāng)前容器緩存中沒(méi)有相對(duì)應(yīng)的 BeanDefinition 對(duì)象,則會(huì)嘗試從父類(lèi)工廠(parentBeanFactory)中加載,然后再去遞歸調(diào)用 getBean(...) 方法

      2.5、標(biāo)記bean為已創(chuàng)建或即將創(chuàng)建

      對(duì)應(yīng)代碼如下:

      //創(chuàng)建的Bean是否需要進(jìn)行類(lèi)型驗(yàn)證,一般不需要
      			if (!typeCheckOnly) {
      				//向容器標(biāo)記指定的Bean已經(jīng)被創(chuàng)建
      				markBeanAsCreated(beanName);
      			}

      typeCheckOnly是doGetBean(final String name, @Nullable final Class requiredType,@Nullable final Object[] args, boolean typeCheckOnly)方法中的一個(gè)參數(shù),一般這個(gè)參數(shù)傳的都是false

      接著追蹤markBeanAsCreated()方法:

      protected void markBeanAsCreated(String beanName) {
      		// 沒(méi)有創(chuàng)建
      		if (!this.alreadyCreated.contains(beanName)) {
      			synchronized (this.mergedBeanDefinitions) {
      				// 再次檢查一次:DCL 雙重校驗(yàn)
      				if (!this.alreadyCreated.contains(beanName)) {
      					clearMergedBeanDefinition(beanName);
      					// 添加到已創(chuàng)建 bean 集合中
      					this.alreadyCreated.add(beanName);
      				}
      			}
      		}
      	}

      這里用到了單例模式中耳熟能詳?shù)碾p重校驗(yàn)

      2.6、獲取BeanDefinition

      對(duì)應(yīng)代碼如下:

              //從容器中獲取 beanName 相應(yīng)的 GenericBeanDefinition 對(duì)象,并將其轉(zhuǎn)換為 RootBeanDefinition 對(duì)象
      	//主要解決Bean繼承時(shí)子類(lèi)合并父類(lèi)公共屬性問(wèn)題
      	final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
      	// 檢查給定的合并的 BeanDefinition (是否為抽象類(lèi))
      	checkMergedBeanDefinition(mbd, beanName, args);

      這段代碼注釋很詳細(xì),就不多解釋了。

      2.7、bean依賴(lài)處理

      對(duì)應(yīng)代碼如下:

      // Guarantee initialization of beans that the current bean depends on.
      	// 處理所依賴(lài)的 bean @DependsOn()
      	//獲取當(dāng)前Bean所有依賴(lài)Bean的名稱(chēng)
      <1>	String[] dependsOn = mbd.getDependsOn();
      	//如果有依賴(lài)
      	if (dependsOn != null) {
      		for (String dep : dependsOn) {
      			//校驗(yàn)該依賴(lài)是否已經(jīng)注冊(cè)過(guò)給當(dāng)前 Bean
      <2>			if (isDependent(beanName, dep)) {
      				//已注冊(cè),拋出異常
      				throw new BeanCreationException(mbd.getResourceDescription(), beanName,
      						"Circular depends-on relationship between '" + beanName + "' and '" + dep + "'");
      			}
      			//沒(méi)有,則先注冊(cè)依賴(lài)的bean
      <3>			registerDependentBean(dep, beanName);
      			//遞歸調(diào)用getBean(),先生成依賴(lài)的bean
      <4>			getBean(dep);
      		}
      	}

      在spring中有一個(gè)@DependsOn注解,它的作用是依賴(lài)加載,比如A對(duì)象要在B對(duì)象加載之后才能加載,那么可以在A上面加@DependsOn(value = "B")注解,就可以達(dá)到我們的要求。
      其實(shí)@DependsOn實(shí)現(xiàn)的原理就是上面這段代碼。

      • <1>、通過(guò)我們前面從IoC容器中拿到的BeanDefinition,調(diào)用mbd.getDependsOn()方法,獲取當(dāng)前bean所有的依賴(lài)。

      • <2>、遍歷這些依賴(lài),判斷此依賴(lài)是否已注冊(cè)給當(dāng)前的Bean

      • <3>、沒(méi)有,則先注冊(cè)依賴(lài)的Bean

      • <4>、遞歸調(diào)用getBean(),先生成依賴(lài)的bean

      <2>、遍歷這些依賴(lài),判斷此依賴(lài)是否已注冊(cè)給當(dāng)前的Bean

      代碼:

      	
      	// 保存的是bean與其依賴(lài)的映射關(guān)系:B - > A
      	private final Map> dependentBeanMap = new ConcurrentHashMap<>(64);
      
              //保存的是bean與其依賴(lài)的映射關(guān)系:A - > B
      	private final Map> dependenciesForBeanMap = new ConcurrentHashMap<>(64);
      
      private boolean isDependent(String beanName, String dependentBeanName, @Nullable Set alreadySeen) {
      		if (alreadySeen != null && alreadySeen.contains(beanName)) {
      			return false;
      		}
      		// 獲取當(dāng)前原始 beanName
      		String canonicalName = canonicalName(beanName);
      		// 獲取該bean依賴(lài)的其他bean集合
      		Set dependentBeans = this.dependentBeanMap.get(canonicalName);
      		if (dependentBeans == null) {
      			return false;
      		}
      		// 存在,則證明該依賴(lài)已經(jīng)注冊(cè)到bean中
      		if (dependentBeans.contains(dependentBeanName)) {
      			return true;
      		}
      		// 遞歸檢測(cè)依賴(lài)
      		for (String transitiveDependency : dependentBeans) {
      			if (alreadySeen == null) {
      				alreadySeen = new HashSet<>();
      			}
      			alreadySeen.add(beanName);
      			if (isDependent(transitiveDependency, dependentBeanName, alreadySeen)) {
      				return true;
      			}
      		}
      		return false;
      	}

      這段代碼很簡(jiǎn)單,主要就是通過(guò)dependentBeanMap獲取當(dāng)前bean對(duì)應(yīng)的所有依賴(lài)dependentBeans,然后判斷是否已注冊(cè),接著遞歸檢查依賴(lài)的Bean有沒(méi)有依賴(lài),如果有,就遞歸調(diào)用isDependent()檢查

      <3>、沒(méi)有,則先注冊(cè)依賴(lài)的Bean

      如果沒(méi)有注冊(cè)依賴(lài)的Bean到該 Bean,則執(zhí)行注冊(cè)registerDependentBean(dep, beanName)

      	// 保存的是bean與其依賴(lài)的映射關(guān)系:B - > A
      	private final Map> dependentBeanMap = new ConcurrentHashMap<>(64);
      
              //保存的是bean與其依賴(lài)的映射關(guān)系:A - > B
      	private final Map> dependenciesForBeanMap = new ConcurrentHashMap<>(64);
      
      //為指定的Bean注入依賴(lài)的Bean
      	public void registerDependentBean(String beanName, String dependentBeanName) {
      	// A quick check for an existing entry upfront, avoiding synchronization...
      	//獲取原始beanName
      	String canonicalName = canonicalName(beanName);
      	Set dependentBeans = this.dependentBeanMap.get(canonicalName);
      	if (dependentBeans != null && dependentBeans.contains(dependentBeanName)) {
      		return;
      	}
      
      	// No entry yet -> fully synchronized manipulation of the dependentBeans Set
      	//先從容器中:bean名稱(chēng)-->全部依賴(lài)Bean名稱(chēng)集合找查找給定名稱(chēng)Bean的依賴(lài)Bean
      	synchronized (this.dependentBeanMap) {
      		//獲取給定名稱(chēng)Bean的所有依賴(lài)Bean名稱(chēng)
      		dependentBeans = this.dependentBeanMap.get(canonicalName);
      		if (dependentBeans == null) {
      			//為Bean設(shè)置依賴(lài)Bean信息
      			dependentBeans = new LinkedHashSet<>(8);
      			this.dependentBeanMap.put(canonicalName, dependentBeans);
      		}
      		//把映射關(guān)系存入集合
      		dependentBeans.add(dependentBeanName);
      	}
      	//從容器中:bean名稱(chēng)-->指定名稱(chēng)Bean的依賴(lài)Bean集合找查找給定名稱(chēng)Bean的依賴(lài)Bean
      	synchronized (this.dependenciesForBeanMap) {
      		Set dependenciesForBean = this.dependenciesForBeanMap.get(dependentBeanName);
      		if (dependenciesForBean == null) {
      			dependenciesForBean = new LinkedHashSet<>(8);
      			this.dependenciesForBeanMap.put(dependentBeanName, dependenciesForBean);
      		}
      		//把映射關(guān)系存入集合
      		dependenciesForBean.add(canonicalName);
      	}
      	}

      套用上面的例子,如果 A@DependsOn(value = "B") ,也就是說(shuō)A依賴(lài)于B,那么該方法registerDependentBean(dep, beanName)中,參數(shù) dep 就是B,beanName 就是A。
      這段代碼中其實(shí)就是把bean之間的依賴(lài)關(guān)系注冊(cè)到兩個(gè)map中。

      • dependentBeanMap 存入(B,A)

      • dependenciesForBeanMap 存入(A,B)

      <4>、遞歸調(diào)用getBean(dep),先生成依賴(lài)的bean

      到了這一步,遞歸調(diào)用getBean(beanName)方法也就是doGetBean(beanName)重走當(dāng)前流程,來(lái)先實(shí)例化依賴(lài)的Bean。等依賴(lài)的Bean實(shí)例化之后,當(dāng)前bean再接著往下執(zhí)行。

      2.8、不同作用域bean的實(shí)例化

      代碼:

              // Create bean instance.
      	//創(chuàng)建單例Bean
      	if (mbd.isSingleton()) {
      		//這里使用了一個(gè)匿名內(nèi)部類(lèi),創(chuàng)建Bean實(shí)例對(duì)象,并且注冊(cè)給所依賴(lài)的對(duì)象
      		sharedInstance = getSingleton(beanName, () -> {
      			try {
      				//創(chuàng)建一個(gè)指定Bean實(shí)例對(duì)象,如果有父級(jí)繼承,則合并子類(lèi)和父類(lèi)的定義
      				return createBean(beanName, mbd, args);
      			}
      			catch (BeansException ex) {
      				// Explicitly remove instance from singleton cache: It might have been put there
      				// eagerly by the creation process, to allow for circular reference resolution.
      				// Also remove any beans that received a temporary reference to the bean.
      				//顯式地從容器單例模式Bean緩存中清除實(shí)例對(duì)象
      				destroySingleton(beanName);
      				throw ex;
      			}
      		});
      		//獲取給定Bean的實(shí)例對(duì)象
      		bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
      	}
      
      	//創(chuàng)建多例Bean
      	else if (mbd.isPrototype()) {
      		//原型模式(Prototype)是每次都會(huì)創(chuàng)建一個(gè)新的對(duì)象
      		Object prototypeInstance = null;
      		try {
      			//加載前置處理,默認(rèn)的功能是注冊(cè)當(dāng)前創(chuàng)建的原型對(duì)象
      			beforePrototypeCreation(beanName);
      			//創(chuàng)建指定Bean對(duì)象實(shí)例
      			prototypeInstance = createBean(beanName, mbd, args);
      		}
      		finally {
      			//加載后置處理,默認(rèn)的功能告訴IOC容器指定Bean的原型對(duì)象不再創(chuàng)建
      			afterPrototypeCreation(beanName);
      		}
      		//獲取給定Bean的實(shí)例對(duì)象
      		bean = getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
      	}
      
      	//要?jiǎng)?chuàng)建的Bean既不是Singleton也不是Prototype
      	//如:request、session、application等生命周期
      	else {
      		String scopeName = mbd.getScope();
      		final Scope scope = this.scopes.get(scopeName);
      		//Bean定義資源中沒(méi)有配置生命周期范圍,則Bean定義不合法
      		if (scope == null) {
      			throw new IllegalStateException("No Scope registered for scope name '" + scopeName + "'");
      		}
      		try {
      			//這里又使用了一個(gè)匿名內(nèi)部類(lèi),獲取一個(gè)指定生命周期范圍的實(shí)例
      			Object scopedInstance = scope.get(beanName, () -> {
      				//前置處理
      				beforePrototypeCreation(beanName);
      				try {
      					return createBean(beanName, mbd, args);
      				}
      				finally {
      					//后置處理
      					afterPrototypeCreation(beanName);
      				}
      			});
      			//獲取給定Bean的實(shí)例對(duì)象
      			bean = getObjectForBeanInstance(scopedInstance, name, beanName, mbd);
      		}
      		catch (IllegalStateException ex) {
      			throw new BeanCreationException(beanName,
      					"Scope '" + scopeName + "' is not active for the current thread; consider " +
      					"defining a scoped proxy for this bean if you intend to refer to it from a singleton",
      					ex);
      		}
      	}

      這段代碼很明顯,分成了3個(gè)部分:

      • singleton Bean實(shí)例化

      • Prototype Bean實(shí)例化

      • 其他類(lèi)型 Bean 實(shí)例化(session,request等)

      我們先來(lái)看singleton Bean實(shí)例化:

      if (mbd.isSingleton()) {
      	//這里使用了一個(gè)匿名內(nèi)部類(lèi),創(chuàng)建Bean實(shí)例對(duì)象,并且注冊(cè)給所依賴(lài)的對(duì)象
      	sharedInstance = getSingleton(beanName, () -> {
      	try {
      		//創(chuàng)建一個(gè)指定Bean實(shí)例對(duì)象,如果有父級(jí)繼承,則合并子類(lèi)和父類(lèi)的定義
      		return createBean(beanName, mbd, args);
      	}
      	catch (BeansException ex) {
      		// Explicitly remove instance from singleton cache: It might have been put there
      		// eagerly by the creation process, to allow for circular reference resolution.
      		// Also remove any beans that received a temporary reference to the bean.
      		//顯式地從容器單例模式Bean緩存中清除實(shí)例對(duì)象
      		destroySingleton(beanName);
      		throw ex;
      	}
      	});
      	//獲取給定Bean的實(shí)例對(duì)象
      	bean = getObjectForBeanInstance(sharedInstance, name,beanName, mbd);
      	}

      Spring Bean 的作用域默認(rèn)為 singleton 。還有其他作用域,如 prototype、request、session 等。
      不同的作用域會(huì)有不同的初始化策略。
      詳見(jiàn)Spring Ioc 之 Bean的加載(三):各個(gè) scope 的 Bean 創(chuàng)建。

      2.9、類(lèi)型轉(zhuǎn)換

      代碼:

      // Check if required type matches the type of the actual bean instance.
      	//對(duì)創(chuàng)建的Bean實(shí)例對(duì)象進(jìn)行類(lèi)型檢查
      	if (requiredType != null && !requiredType.isInstance(bean)) {
      	try {
      	        //執(zhí)行轉(zhuǎn)換
      		T convertedBean = getTypeConverter().convertIfNecessary(bean, requiredType);
      		// 轉(zhuǎn)換失敗,拋異常
      		if (convertedBean == null) {
      			throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
      		}
      		return convertedBean;
      	}
      	catch (TypeMismatchException ex) {
      		if (logger.isDebugEnabled()) {
      			logger.debug("Failed to convert bean '" + name + "' to required type '" +
      					ClassUtils.getQualifiedName(requiredType) + "'", ex);
      		}
      		throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
      	}
      	}
      	return (T) bean;

      requiredTypegetBean()方法可傳入的一個(gè)參數(shù),即可以根據(jù)指定的 beanName 和 requiredType 來(lái)獲取Bean。

      但是一般情況下是不需要類(lèi)型檢查的,requiredType一般為null,如getBean(beanName)

      當(dāng)requiredType不為null的時(shí)候走這段邏輯。

      關(guān)于Spring Ioc中怎么加載Bean就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到。


      網(wǎng)頁(yè)名稱(chēng):SpringIoc中怎么加載Bean
      標(biāo)題網(wǎng)址:http://www.ef60e0e.cn/article/gcdjos.html 99热在线精品一区二区三区_国产伦精品一区二区三区女破破_亚洲一区二区三区无码_精品国产欧美日韩另类一区
          1. <ul id="0c1fb"></ul>

            <noscript id="0c1fb"><video id="0c1fb"></video></noscript>
            <noscript id="0c1fb"><listing id="0c1fb"><thead id="0c1fb"></thead></listing></noscript>

            准格尔旗| 阳泉市| 封开县| 甘谷县| 镇坪县| 通山县| 三亚市| 始兴县| 玛曲县| 尚志市| 商丘市| 揭阳市| 永德县| 徐闻县| 通化市| 泽州县| 兴宁市| 巴青县| 亚东县| 万荣县| 桃江县| 吉木萨尔县| 米林县| 浑源县| 汉沽区| 清新县| 柳江县| 虞城县| 彭州市| 礼泉县| 卢龙县| 漯河市| 桃江县| 抚宁县| 稻城县| 玉屏| 汝州市| 清新县| 微山县| 奉新县| 焦作市|