日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 前端技术 > javascript >内容正文

javascript

Spring IoC容器设计原理及高级特性

發布時間:2023/12/18 javascript 64 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Spring IoC容器设计原理及高级特性 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

文章目錄

    • Spring IoC容器概述
    • IoC容器系列的設計與實現:BeanFactory和ApplicationContext
      • BeanFactory
      • BeanFactory容器的設計原理
      • ApplicationContext
      • ApplicationContext容器的設計原理
    • IoC容器的初始化過程
    • IoC容器的依賴注入
      • 依賴注入發生的時間
      • 依賴注入的流程圖
      • 依賴注入的源碼分析
    • 容器其他相關特性的設計與實現
      • Application中Bean的初始化及銷毀
      • lazy-init屬性和預實例化
      • FactoryBean的實現
      • BeanPostProcessor的實現
      • autowiring(自動裝配的實現)
      • Bean的依賴檢查

Spring IoC容器概述

IoC是Spring容器的內核,其字面意思是控制反轉。那么究竟是哪些方面的控制被反轉了?其實是依賴對象的獲得被反轉了。基于此,提出了DI(依賴注入)的概念。

IoC容器系列的設計與實現:BeanFactory和ApplicationContext

在Spring IoC容器的設計中,主要有兩個容器系列,一個是實現BeanFactory接口餓簡單容器系列,這系列容器只實現了容器的最基本功能;另一個是ApplicationContext應用上下文,它作為容器的高級形態而存在。應用上下文在簡單容器的基礎上,增加了許多面向框架的特性,同時對應用環境做了許多適配。

Spring通過定義BeanDefinition來管理基于Spring的應用中的各種對象以及它們之間的相互依賴關系。對IoC容器來說,BeanDefinition就是依賴反轉模式中管理的對象依賴關系的數據抽象,也是容器實現依賴反轉功能的核心數據結構,依賴反轉功能都是圍繞這個BeanDefinition的處理來完成的。

? 從接口BeanFactory到HierarchicalBeanFactory,再到ConfigurableBeanFactory,是一條主要的BeanFactory設計路徑
BeanFactory定義了基本的IoC容器的規范。包括了getBean()(通過這個方法可以從容器中取得Bean)。
HierarchicalBeanFactory接口在繼承了BeanFactory后,增加了getParentBeanFactory(),使BeanFactory具備了雙親IoC容器的管理功能。
在接下來的ConfigurableBeanFactory中,定義了一些對BeanFactory的配置功能,比如通過setParentBeanFactory()設置雙親IoC容器,通過addBeanPostProcessor()配置Bean后置處理器,等等。

? 第二條接口設計主線是,以ApplicationContext為核心的接口設計
我們常用的應用上下文基本上都是ConfigurableApplicationContext或者WebApplicationContext的實現
在這個接口體系中,ListableBeanFactory和HierarchicalBeanFactory兩個接口,連接BeanFactory接口定義和ApplicationConext應用上下文的接口定義。
在ListableBeanFactory接口中,細化了許多BeanFactory的接口功能,比如定義了getBeanDefinitionNames()接口方法;對于HierarchicalBeanFactory接口,我們在前文中已經提到過;對于ApplicationContext接口,它通過繼承MessageSource、ResourceLoader、ApplicationEventPublisher接口,在BeanFactory簡單IoC容器的基礎上添加了許多對高級容器的特性的支持

? 這里涉及的是主要接口關系,而具體的IoC容器都是在這個接口體系下實現的,比如DefaultListableBeanFactory,這個基本IoC容器的實現就是實現了ConfigurableBeanFactory,從而成為一個簡單IoC容器的實現。
像其他IoC容器,比如XmlBeanFactory,都是在DefaultListableBeanFactory的基礎上做擴展

? 這個接口系統是以BeanFactory和ApplicationContext為核心
而BeanFactory又是IoC容器的最基本接口,在ApplicationContext的設計中,一方面,可以看到它繼承了BeanFactory接口體系的接口,具備了BeanFactory IoC容器的基本功能
另一方面,通過繼承MessageSource、ResourceLoadr、ApplicationEventPublisher這些接口,BeanFactory為ApplicationContext賦予了更高級的IoC容器特性。
對于ApplicationContext而言,為了在Web環境中使用它,還設計了WebApplicationContext接口,而這個接口通過繼承ThemeSource接口來擴充功能。

BeanFactory

Spring Bean的創建是典型的工廠模式,這一系列的Bean工廠,也即IOC容器為開發者管理對象間的依賴關系提供了很多便利和基礎服務,在Spring中有許多的IOC容器的實現供用戶選擇和使用,其相互關系如下:

BeanFactory接口定義了IoC容器最基本的形式,不關心 Bean 是怎樣定義和加載的。如果我們想要知道一個工廠具體產生對象的過程,則要看這個接口的實現類。

其中BeanFactory作為最頂層的一個接口類,它定義了IOC容器的基本功能規范,BeanFactory 有三個子類:
ListableBeanFactory、HierarchicalBeanFactory 和AutowireCapableBeanFactory。

但是從上圖中我們可以發現最終的默認實現類是 DefaultListableBeanFactory,他實現了所有的接口。那為何要定義這么多層次的接口呢?查閱這些接口的源碼和說明發現,每個接口都有他使用的場合,它主要是為了區分在 Spring 內部在操作過程中對象的傳遞和轉化過程中,對對象的數據訪問所做的限制。例如 ListableBeanFactory 接口表示這些 Bean 是可列表的,而 HierarchicalBeanFactory 表示子容器可以通過接口方法訪問父容器,AutowireCapableBeanFactory 接口定義 Bean 的自動裝配規則。這四個接口共同定義了 Bean 的集合、Bean 之間的關系、以及 Bean 行為。

最基本的IOC容器接口BeanFactory

public interface BeanFactory { //對FactoryBean的轉義定義,因為如果使用bean的名字檢索FactoryBean得到的對象是工廠生成的對象, //如果需要得到工廠本身,需要轉義 String FACTORY_BEAN_PREFIX = "&"; //根據bean的名字,獲取在IOC容器中得到bean實例 Object getBean(String name) throws BeansException; //根據bean的名字和Class類型來得到bean實例,增加了類型安全驗證機制。 Object getBean(String name, Class requiredType) throws BeansException; //提供對bean的檢索,看看是否在IOC容器有這個名字的bean boolean containsBean(String name); //根據bean名字得到bean實例,并同時判斷這個bean是不是單例 boolean isSingleton(String name) throws NoSuchBeanDefinitionException; //得到bean實例的Class類型 Class getType(String name) throws NoSuchBeanDefinitionException; //得到bean的別名,如果根據別名檢索,那么其原名也會被檢索出來 String[] getAliases(String name); }

BeanFactory容器的設計原理

在BeanFactory接口的基礎上,Spring提供了一系列容器的實現供開發人員使用。我們以XmlBeanFactory的實現為例來說明簡單IoC容器的設計原理。

XmlBeanFactory繼承自DefaultListableBeanFactory這個類,同時實現了其他諸如XML讀取的附加功能。即它是一個可以讀取以XML文件方式定義的BeanDefinition的IoC容器。

public class XmlBeanFactory extends DefaultListableBeanFactory{private final XmlBeanDefinitionReader reader; public XmlBeanFactory(Resource resource)throws BeansException{this(resource, null);}public XmlBeanFactory(Resource resource, BeanFactory parentBeanFactory)throws BeansException{super(parentBeanFactory);this.reader = new XmlBeanDefinitionReader(this);this.reader.loadBeanDefinitions(resource);}}

在XmlBeanFactory中,初始化了一個XmlBeanDefinitionReader對象,由它來完成XML形式的信息處理。構造XmlBeanFactory這個容器時,需要指定BeanDefinition的信息來源,將它封裝成Spring中的Resource類來給出。然后傳遞給XmlBeanFactory構造函數,IoC容器就可以方便地定位到需要的BeanDefinition信息來對Bean完成容器的初始化和依賴注入過程。對XmlBeanDefinitionReader對象的初始化,以及使用這個對象來完成對loadBeanDefinition的調用,就是這個調用啟動從Resource中載入BeanDefinitions的過程。

ApplicationContext

ApplicationContext是Spring提供的一個高級的IoC容器,它除了能夠提供IoC容器的基本功能外,還為用戶提供了以下的附加服務。

  • 支持信息源,可以實現國際化。(實現MessageSource接口)
  • 訪問資源。(實現ResourcePatternResolver接口,這個后面要講)
  • 支持應用事件。(實現ApplicationEventPublisher接口)
  • 在ApplicationContext中提供的附加服務
  • ApplicationContext容器的設計原理

    我們以常用的FileSystemXmlApplicationContext的實現為例說明ApplicationContext容器的設計原理。

    public FileSystemXmlApplicationContext(String[] configLocations, boolean refresh, ApplicationContext parent) throws BeansException { super(parent); setConfigLocations(configLocations); if (refresh) { refresh(); } }

    在FileSystemXmlApplicationContext的設計中,我們看到ApplicationContext應用上下文的主要功能已經在FileSystemXmlApplicationContext的基類AbstractXmlApplicationContext中實現了,FileSystemXmlApplicationContext作為一個具體的應用上下文,只需要實現和它自身設計相關的兩個功能。一個功能是,如果應用直接使用FileSystemXmlApplicationContext,對于實例化這個應用上下文的支持,同時啟動IoC容器的refresh()過程。另一個功能是與FileSystemXmlApplicationContext設計具體相關的功能,這部分與怎樣從文件系統中加載XML的Bean定義資源有關。

    IoC容器的初始化過程

    IoC容器的初始化是由refresh()方法來啟動的,這個方法標志著容器的正式啟動。具體來說,這個啟動包括BeanDefinition的Resource定位、載入和注冊三個基本過程。

    • Resource定位:指的是BeanDefinition的資源定位,它由ResourceLoader通過統一的Resource接口來完成。
    • BeanDefinition的載入:把用戶定義好的Bea表示成IoC容器內部的數據結構,而這個容器內部的數據結構就是BeanDefinition。
    • BeanDefinition的注冊:通過調用BeanDefinitionRegistry接口的實現來完成的。這個注冊過程是把載入過程中解析得到的BeanDefinition向IoC容器進行注冊。

    IoC容器的依賴注入

    依賴注入發生的時間

    當Spring IoC容器完成了Bean定義資源的定位、載入和解析注冊以后,IoC容器中已經管理類Bean定義的相關數據,即初始化過程完成的主要工作是在IoC容器中建立BeanDefinition數據映射。
    但是此時IoC容器還沒有對所管理的Bean進行依賴注入,依賴注入在以下兩種情況發生:

    (1) 用戶第一次通過getBean方法向IoC容索要Bean時,IoC容器觸發依賴注入。

    (2) 當用戶在Bean定義資源中為< Bean >元素配置了lazy-init屬性,即讓容器在解析注冊Bean定義時進行預實例化,觸發依賴注入。

    BeanFactory接口定義了Spring IoC容器的基本功能規范,是Spring IoC容器所應遵守的最底層和最基本的編程規范。BeanFactory接口中定義了幾個getBean方法,就是用戶向IoC容器索取管理的Bean的方法,我們通過分析其子類的具體實現,理解Spring IoC容器在用戶索取Bean時如何完成依賴注入。

    依賴注入的流程圖

    依賴注入的源碼分析

    在BeanFactory中我們看到getBean()函數,它的具體實現在AbstractBeanFactory中。

  • AbstractBeanFactory通過getBean向IoC容器獲取被管理的Bean。
  • AbstractBeanFactory的getBean相關方法的源碼如下:

    //獲取IoC容器中指定名稱的Bean public Object getBean(String name) throws BeansException { //doGetBean才是真正向IoC容器獲取被管理Bean的過程 return doGetBean(name, null, null, false); } //獲取IoC容器中指定名稱和類型的Bean public <T> T getBean(String name, Class<T> requiredType) throws BeansException { //doGetBean才是真正向IoC容器獲取被管理Bean的過程 return doGetBean(name, requiredType, null, false); } //獲取IoC容器中指定名稱和參數的Bean public Object getBean(String name, Object... args) throws BeansException { //doGetBean才是真正向IoC容器獲取被管理Bean的過程 return doGetBean(name, null, args, false); } //獲取IoC容器中指定名稱、類型和參數的Bean public <T> T getBean(String name, Class<T> requiredType, Object... args) throws BeansException { //doGetBean才是真正向IoC容器獲取被管理Bean的過程 return doGetBean(name, requiredType, args, false); } //真正實現向IoC容器獲取Bean的功能,也是觸發依賴注入功能的地方 @SuppressWarnings("unchecked") protected <T> T doGetBean( final String name, final Class<T> requiredType, final Object[] args, boolean typeCheckOnly) throws BeansException { //根據指定的名稱獲取被管理Bean的名稱,剝離指定名稱中對容器的相關依賴 //如果指定的是別名,將別名轉換為規范的Bean名稱 final String beanName = transformedBeanName(name); Object bean; //先從緩存中取是否已經有被創建過的單例類型的Bean,對于單例模式的Bean整 //個IoC容器中只創建一次,不需要重復創建 Object sharedInstance = getSingleton(beanName); //IoC容器創建單例模式Bean實例對象 if (sharedInstance != null && args == null) { if (logger.isDebugEnabled()) { //如果指定名稱的Bean在容器中已有單例模式的Bean被創建,直接返回 //已經創建的Bean 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 + "'"); } } //獲取給定Bean的實例對象,主要是完成FactoryBean的相關處理 //注意:BeanFactory是管理容器中Bean的工廠,而FactoryBean是 //創建創建對象的工廠Bean,兩者之間有區別 bean = getObjectForBeanInstance(sharedInstance, name, beanName, null); } else {//緩存沒有正在創建的單例模式Bean //緩存中已經有已經創建的原型模式Bean,但是由于循環引用的問題導致實 //例化對象失敗 if (isPrototypeCurrentlyInCreation(beanName)) { throw new BeanCurrentlyInCreationException(beanName); } //對IoC容器中是否存在指定名稱的BeanDefinition進行檢查,首先檢查是否 //能在當前的BeanFactory中獲取的所需要的Bean,如果不能則委托當前容器 //的父級容器去查找,如果還是找不到則沿著容器的繼承體系向父級容器查找 BeanFactory parentBeanFactory = getParentBeanFactory(); //當前容器的父級容器存在,且當前容器中不存在指定名稱的Bean if (parentBeanFactory != null && !containsBeanDefinition(beanName)) { //解析指定Bean名稱的原始名稱 String nameToLookup = originalBeanName(name); if (args != null) { //委派父級容器根據指定名稱和顯式的參數查找 return (T) parentBeanFactory.getBean(nameToLookup, args); } else { //委派父級容器根據指定名稱和類型查找 return parentBeanFactory.getBean(nameToLookup, requiredType); } } //創建的Bean是否需要進行類型驗證,一般不需要 if (!typeCheckOnly) { //向容器標記指定的Bean已經被創建 markBeanAsCreated(beanName); } //根據指定Bean名稱獲取其父級的Bean定義,主要解決Bean繼承時子類 //合并父類公共屬性問題 final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName); checkMergedBeanDefinition(mbd, beanName, args); //獲取當前Bean所有依賴Bean的名稱 String[] dependsOn = mbd.getDependsOn(); //如果當前Bean有依賴Bean if (dependsOn != null) { for (String dependsOnBean : dependsOn) { //遞歸調用getBean方法,獲取當前Bean的依賴Bean getBean(dependsOnBean); //把被依賴Bean注冊給當前依賴的Bean registerDependentBean(dependsOnBean, beanName); } } //創建單例模式Bean的實例對象 if (mbd.isSingleton()) { //這里使用了一個匿名內部類,創建Bean實例對象,并且注冊給所依賴的對象 sharedInstance = getSingleton(beanName, new ObjectFactory() { public Object getObject() throws BeansException { try { //創建一個指定Bean實例對象,如果有父級繼承,則合并子//類和父類的定義 return createBean(beanName, mbd, args); } catch (BeansException ex) { //顯式地從容器單例模式Bean緩存中清除實例對象 destroySingleton(beanName); throw ex; } } }); //獲取給定Bean的實例對象 bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd); } //IoC容器創建原型模式Bean實例對象 else if (mbd.isPrototype()) { //原型模式(Prototype)是每次都會創建一個新的對象 Object prototypeInstance = null; try { //回調beforePrototypeCreation方法,默認的功能是注冊當前創//建的原型對象 beforePrototypeCreation(beanName); //創建指定Bean對象實例 prototypeInstance = createBean(beanName, mbd, args); } finally { //回調afterPrototypeCreation方法,默認的功能告訴IoC容器指//定Bean的原型對象不再創建了 afterPrototypeCreation(beanName); } //獲取給定Bean的實例對象 bean = getObjectForBeanInstance(prototypeInstance, name, beanName, mbd); } //要創建的Bean既不是單例模式,也不是原型模式,則根據Bean定義資源中 //配置的生命周期范圍,選擇實例化Bean的合適方法,這種在Web應用程序中 //比較常用,如:request、session、application等生命周期 else { String scopeName = mbd.getScope(); final Scope scope = this.scopes.get(scopeName); //Bean定義資源中沒有配置生命周期范圍,則Bean定義不合法 if (scope == null) { throw new IllegalStateException("No Scope registered for scope '" + scopeName + "'"); } try { //這里又使用了一個匿名內部類,獲取一個指定生命周期范圍的實例 Object scopedInstance = scope.get(beanName, new ObjectFactory() { public Object getObject() throws BeansException { beforePrototypeCreation(beanName); try { return createBean(beanName, mbd, args); } finally { afterPrototypeCreation(beanName); } } }); //獲取給定Bean的實例對象 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); } } } //對創建的Bean實例對象進行類型檢查 if (requiredType != null && bean != null && !requiredType.isAssignableFrom(bean.getClass())) { throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass()); } return (T) bean; }

    通過上面對向IoC容器獲取Bean方法的分析,我們可以看到在Spring中,如果Bean定義的單例模式(Singleton),則容器在創建之前先從緩存中查找,以確保整個容器中只存在一個實例對象。如果Bean定義的是原型模式(Prototype),則容器每次都會創建一個新的實例對象。除此之外,Bean定義還可以擴展為指定其生命周期范圍。

    上面的源碼只是定義了根據Bean定義的模式,采取的不同創建Bean實例對象的策略,具體的Bean實例對象的創建過程由實現了ObejctFactory接口的匿名內部類的createBean方法完成,ObejctFactory使用委派模式,具體的Bean實例創建過程交由其實現類AbstractAutowireCapableBeanFactory完成,我們繼續分析AbstractAutowireCapableBeanFactory的createBean方法的源碼,理解其創建Bean實例的具體實現過程。

  • AbstractAutowireCapableBeanFactory創建Bean實例對象。
  • AbstractAutowireCapableBeanFactory類實現了ObejctFactory接口,創建容器指定的Bean實例對象,同時還對創建的Bean實例對象進行初始化處理。其創建Bean實例對象的方法源碼如下:

    //創建Bean實例對象 protected Object createBean(final String beanName, final RootBeanDefinition mbd, final Object[] args) throws BeanCreationException { if (logger.isDebugEnabled()) { logger.debug("Creating instance of bean '" + beanName + "'"); } //判斷需要創建的Bean是否可以實例化,即是否可以通過當前的類加載器加載 resolveBeanClass(mbd, beanName); //校驗和準備Bean中的方法覆蓋 try { mbd.prepareMethodOverrides(); } catch (BeanDefinitionValidationException ex) { throw new BeanDefinitionStoreException(mbd.getResourceDescription(), beanName, "Validation of method overrides failed", ex); } try { //如果Bean配置了初始化前和初始化后的處理器,則試圖返回一個需要創建//Bean的代理對象 Object bean = resolveBeforeInstantiation(beanName, mbd); if (bean != null) { return bean; } } catch (Throwable ex) { throw new BeanCreationException(mbd.getResourceDescription(), beanName, "BeanPostProcessor before instantiation of bean failed", ex); } //創建Bean的入口 Object beanInstance = doCreateBean(beanName, mbd, args); if (logger.isDebugEnabled()) { logger.debug("Finished creating instance of bean '" + beanName + "'"); } return beanInstance; } //真正創建Bean的方法 protected Object doCreateBean(final String beanName, final RootBeanDefinition mbd, final Object[] args) { //封裝被創建的Bean對象 BeanWrapper instanceWrapper = null; if (mbd.isSingleton()){//單例模式的Bean,先從容器中緩存中獲取同名Bean instanceWrapper = this.factoryBeanInstanceCache.remove(beanName); } if (instanceWrapper == null) { //創建實例對象 instanceWrapper = createBeanInstance(beanName, mbd, args); } final Object bean = (instanceWrapper != null ? instanceWrapper.getWrappedInstance() : null); //獲取實例化對象的類型 Class beanType = (instanceWrapper != null ? instanceWrapper.getWrappedClass() : null); //調用PostProcessor后置處理器 synchronized (mbd.postProcessingLock) { if (!mbd.postProcessed) { applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName); mbd.postProcessed = true; } } // Eagerly cache singletons to be able to resolve circular references //向容器中緩存單例模式的Bean對象,以防循環引用 boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences && isSingletonCurrentlyInCreation(beanName)); if (earlySingletonExposure) { if (logger.isDebugEnabled()) { logger.debug("Eagerly caching bean '" + beanName + "' to allow for resolving potential circular references"); } //這里是一個匿名內部類,為了防止循環引用,盡早持有對象的引用 addSingletonFactory(beanName, new ObjectFactory() { public Object getObject() throws BeansException { return getEarlyBeanReference(beanName, mbd, bean); } }); } //Bean對象的初始化,依賴注入在此觸發 //這個exposedObject在初始化完成之后返回作為依賴注入完成后的Bean Object exposedObject = bean; try { //將Bean實例對象封裝,并且Bean定義中配置的屬性值賦值給實例對象 populateBean(beanName, mbd, instanceWrapper); if (exposedObject != null) { //初始化Bean對象 exposedObject = initializeBean(beanName, exposedObject, mbd); } } catch (Throwable ex) { if (ex instanceof BeanCreationException && beanName.equals(((BeanCreationException) ex).getBeanName())) { throw (BeanCreationException) ex; } else { throw new BeanCreationException(mbd.getResourceDescription(), beanName, "Initialization of bean failed", ex); } } if (earlySingletonExposure) { //獲取指定名稱的已注冊的單例模式Bean對象 Object earlySingletonReference = getSingleton(beanName, false); if (earlySingletonReference != null) { //根據名稱獲取的以注冊的Bean和正在實例化的Bean是同一個 if (exposedObject == bean) { //當前實例化的Bean初始化完成 exposedObject = earlySingletonReference; } //當前Bean依賴其他Bean,并且當發生循環引用時不允許新創建實例對象 else if (!this.allowRawInjectionDespiteWrapping && hasDependentBean(beanName)) { String[] dependentBeans = getDependentBeans(beanName); Set<String> actualDependentBeans = new LinkedHashSet<String>(dependentBeans.length); //獲取當前Bean所依賴的其他Bean for (String dependentBean : dependentBeans) { //對依賴Bean進行類型檢查 if (!removeSingletonIfCreatedForTypeCheckOnly(dependentBean)) { actualDependentBeans.add(dependentBean); } } if (!actualDependentBeans.isEmpty()) { throw new BeanCurrentlyInCreationException(beanName, "Bean with name '" + beanName + "' has been injected into other beans [" + StringUtils.collectionToCommaDelimitedString(actualDependentBeans) + "] in its raw version as part of a circular reference, but has eventually been " + "wrapped. This means that said other beans do not use the final version of the " + "bean. This is often the result of over-eager type matching - consider using " + "'getBeanNamesOfType' with the 'allowEagerInit' flag turned off, for example."); } } } } //注冊完成依賴注入的Bean try { registerDisposableBeanIfNecessary(beanName, bean, mbd); } catch (BeanDefinitionValidationException ex) { throw new BeanCreationException(mbd.getResourceDescription(), beanName, "Invalid destruction signature", ex); } return exposedObject; }

    通過對方法源碼的分析,我們看到具體的依賴注入實現在以下兩個方法中:
    (1) createBeanInstance:生成Bean所包含的java對象實例。

    (2) populateBean :對Bean屬性的依賴注入進行處理。

    下面繼續分析這兩個方法的代碼實現。

  • createBeanInstance方法創建Bean的java實例對象。
  • 在createBeanInstance方法中,根據指定的初始化策略,使用靜態工廠、工廠方法或者容器的自動裝配特性生成java實例對象,創建對象的源碼如下:

    //創建Bean的實例對象 protected BeanWrapper createBeanInstance(String beanName, RootBeanDefinition mbd, Object[] args) { //檢查確認Bean是可實例化的 Class beanClass = resolveBeanClass(mbd, beanName); //使用工廠方法對Bean進行實例化 if (beanClass != null && !Modifier.isPublic(beanClass.getModifiers()) && !mbd.isNonPublicAccessAllowed()) { throw new BeanCreationException(mbd.getResourceDescription(), beanName, "Bean class isn't public, and non-public access not allowed: " + beanClass.getName()); } if (mbd.getFactoryMethodName() != null) { //調用工廠方法實例化 return instantiateUsingFactoryMethod(beanName, mbd, args); } //使用容器的自動裝配方法進行實例化 boolean resolved = false; boolean autowireNecessary = false; if (args == null) { synchronized (mbd.constructorArgumentLock) { if (mbd.resolvedConstructorOrFactoryMethod != null) { resolved = true; autowireNecessary = mbd.constructorArgumentsResolved; } } } if (resolved) { if (autowireNecessary) { //配置了自動裝配屬性,使用容器的自動裝配實例化 //容器的自動裝配是根據參數類型匹配Bean的構造方法 return autowireConstructor(beanName, mbd, null, null); } else { //使用默認的無參構造方法實例化 return instantiateBean(beanName, mbd); } } //使用Bean的構造方法進行實例化 Constructor[] ctors = determineConstructorsFromBeanPostProcessors(beanClass, beanName); if (ctors != null || mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_CONSTRUCTOR || mbd.hasConstructorArgumentValues() || !ObjectUtils.isEmpty(args)) { //使用容器的自動裝配特性,調用匹配的構造方法實例化 return autowireConstructor(beanName, mbd, ctors, args); } //使用默認的無參構造方法實例化 return instantiateBean(beanName, mbd); } //使用默認的無參構造方法實例化Bean對象 protected BeanWrapper instantiateBean(final String beanName, final RootBeanDefinition mbd) { try { Object beanInstance; final BeanFactory parent = this; //獲取系統的安全管理接口,JDK標準的安全管理API if (System.getSecurityManager() != null) { //這里是一個匿名內置類,根據實例化策略創建實例對象 beanInstance = AccessController.doPrivileged(new PrivilegedAction<Object>() { public Object run() { return getInstantiationStrategy().instantiate(mbd, beanName, parent); } }, getAccessControlContext()); } else { //將實例化的對象封裝起來 beanInstance = getInstantiationStrategy().instantiate(mbd, beanName, parent); } BeanWrapper bw = new BeanWrapperImpl(beanInstance); initBeanWrapper(bw); return bw; } catch (Throwable ex) { throw new BeanCreationException(mbd.getResourceDescription(), beanName, "Instantiation of bean failed", ex); } }

    經過對上面的代碼分析,我們可以看出,對使用工廠方法和自動裝配特性的Bean的實例化相對比較清楚,調用相應的工廠方法或者參數匹配的構造方法即可完成實例化對象的工作,但是對于我們最常使用的默認無參構造方法就需要使用相應的初始化策略(JDK的反射機制或者CGLIB)來進行初始化了,在方法getInstantiationStrategy().instantiate中就具體實現類使用初始策略實例化對象。

  • SimpleInstantiationStrategy類使用默認的無參構造方法創建Bean實例化對象。
  • 在使用默認的無參構造方法創建Bean的實例化對象時,方法getInstantiationStrategy().instantiate調用了SimpleInstantiationStrategy類中的實例化Bean的方法,其源碼如下:

    //使用初始化策略實例化Bean對象 public Object instantiate(RootBeanDefinition beanDefinition, String beanName, BeanFactory owner) { //如果Bean定義中沒有方法覆蓋,則就不需要CGLIB父類類的方法 if (beanDefinition.getMethodOverrides().isEmpty()) { Constructor<?> constructorToUse; synchronized (beanDefinition.constructorArgumentLock) { //獲取對象的構造方法或工廠方法 constructorToUse = (Constructor<?>) beanDefinition.resolvedConstructorOrFactoryMethod; //如果沒有構造方法且沒有工廠方法 if (constructorToUse == null) { //使用JDK的反射機制,判斷要實例化的Bean是否是接口 final Class clazz = beanDefinition.getBeanClass(); if (clazz.isInterface()) { throw new BeanInstantiationException(clazz, "Specified class is an interface"); } try { if (System.getSecurityManager() != null) { //這里是一個匿名內置類,使用反射機制獲取Bean的構造方法 constructorToUse = AccessController.doPrivileged(new PrivilegedExceptionAction<Constructor>() { public Constructor run() throws Exception { return clazz.getDeclaredConstructor((Class[]) null); } }); } else { constructorToUse = clazz.getDeclaredConstructor((Class[]) null); } beanDefinition.resolvedConstructorOrFactoryMethod = constructorToUse; } catch (Exception ex) { throw new BeanInstantiationException(clazz, "No default constructor found", ex); } } } //使用BeanUtils實例化,通過反射機制調用”構造方法.newInstance(arg)”來進行實例化 return BeanUtils.instantiateClass(constructorToUse); } else { //使用CGLIB來實例化對象 return instantiateWithMethodInjection(beanDefinition, beanName, owner); } }

    通過上面的代碼分析,我們看到了如果Bean有方法被覆蓋了,則使用JDK的反射機制進行實例化,否則,使用CGLIB進行實例化。

    instantiateWithMethodInjection方法調用SimpleInstantiationStrategy的子類CglibSubclassingInstantiationStrategy使用CGLIB來進行初始化,其源碼如下:

    //使用CGLIB進行Bean對象實例化 public Object instantiate(Constructor ctor, Object[] args) { //CGLIB中的類 Enhancer enhancer = new Enhancer(); //將Bean本身作為其基類 enhancer.setSuperclass(this.beanDefinition.getBeanClass()); enhancer.setCallbackFilter(new CallbackFilterImpl()); enhancer.setCallbacks(new Callback[] { NoOp.INSTANCE, new LookupOverrideMethodInterceptor(), new ReplaceOverrideMethodInterceptor() }); //使用CGLIB的create方法生成實例對象 return (ctor == null) ? enhancer.create() : enhancer.create(ctor.getParameterTypes(), args); }

    CGLIB是一個常用的字節碼生成器的類庫,它提供了一系列API實現java字節碼的生成和轉換功能。我們在學習JDK的動態代理時都知道,JDK的動態代理只能針對接口,如果一個類沒有實現任何接口,要對其進行動態代理只能使用CGLIB。

  • populateBean方法對Bean屬性的依賴注入:
  • 在第2步的分析中我們已經了解到Bean的依賴注入分為以下兩個過程:
    (1) createBeanInstance:生成Bean所包含的java對象實例。

    (2) populateBean :對Bean屬性的依賴注入進行處理。

    第3、4步中我們已經分析了容器初始化生成Bean所包含的Java實例對象的過程,現在我們繼續分析生成對象后,Spring IoC容器是如何將Bean的屬性依賴關系注入Bean實例對象中并設置好的,屬性依賴注入的代碼如下:

    //將Bean屬性設置到生成的實例對象上 protected void populateBean(String beanName, AbstractBeanDefinition mbd, BeanWrapper bw) { //獲取容器在解析Bean定義資源時為BeanDefiniton中設置的屬性值 PropertyValues pvs = mbd.getPropertyValues(); //實例對象為null if (bw == null) { //屬性值不為空 if (!pvs.isEmpty()) { throw new BeanCreationException( mbd.getResourceDescription(), beanName, "Cannot apply property values to null instance"); } else { //實例對象為null,屬性值也為空,不需要設置屬性值,直接返回 return; } } //在設置屬性之前調用Bean的PostProcessor后置處理器 boolean continueWithPropertyPopulation = true; 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; } } } } if (!continueWithPropertyPopulation) { return; } //依賴注入開始,首先處理autowire自動裝配的注入 if (mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_NAME || mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_TYPE) { MutablePropertyValues newPvs = new MutablePropertyValues(pvs); //對autowire自動裝配的處理,根據Bean名稱自動裝配注入 if (mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_NAME) { autowireByName(beanName, mbd, bw, newPvs); } //根據Bean類型自動裝配注入 if (mbd.getResolvedAutowireMode() == RootBeanDefinition.AUTOWIRE_BY_TYPE) { autowireByType(beanName, mbd, bw, newPvs); } pvs = newPvs; } //檢查容器是否持有用于處理單態模式Bean關閉時的后置處理器 boolean hasInstAwareBpps = hasInstantiationAwareBeanPostProcessors(); //Bean實例對象沒有依賴,即沒有繼承基類 boolean needsDepCheck = (mbd.getDependencyCheck() != RootBeanDefinition.DEPENDENCY_CHECK_NONE); if (hasInstAwareBpps || needsDepCheck) { //從實例對象中提取屬性描述符 PropertyDescriptor[] filteredPds = filterPropertyDescriptorsForDependencyCheck(bw); if (hasInstAwareBpps) { for (BeanPostProcessor bp : getBeanPostProcessors()) { if (bp instanceof InstantiationAwareBeanPostProcessor) { InstantiationAwareBeanPostProcessor ibp = (InstantiationAwareBeanPostProcessor) bp; //使用BeanPostProcessor處理器處理屬性值 pvs = ibp.postProcessPropertyValues(pvs, filteredPds, bw.getWrappedInstance(), beanName); if (pvs == null) { return; } } } } if (needsDepCheck) { //為要設置的屬性進行依賴檢查 checkDependencies(beanName, mbd, filteredPds, pvs); } } //對屬性進行注入 applyPropertyValues(beanName, mbd, bw, pvs); } //解析并注入依賴屬性的過程 protected void applyPropertyValues(String beanName, BeanDefinition mbd, BeanWrapper bw, PropertyValues pvs) { if (pvs == null || pvs.isEmpty()) { return; } //封裝屬性值 MutablePropertyValues mpvs = null; List<PropertyValue> original; if (System.getSecurityManager()!= null) { if (bw instanceof BeanWrapperImpl) { //設置安全上下文,JDK安全機制 ((BeanWrapperImpl) bw).setSecurityContext(getAccessControlContext()); } } if (pvs instanceof MutablePropertyValues) { mpvs = (MutablePropertyValues) pvs; //屬性值已經轉換 if (mpvs.isConverted()) { try { //為實例化對象設置屬性值 bw.setPropertyValues(mpvs); return; } catch (BeansException ex) { throw new BeanCreationException( mbd.getResourceDescription(), beanName, "Error setting property values", ex); } } //獲取屬性值對象的原始類型值 original = mpvs.getPropertyValueList(); } else { original = Arrays.asList(pvs.getPropertyValues()); } //獲取用戶自定義的類型轉換 TypeConverter converter = getCustomTypeConverter(); if (converter == null) { converter = bw; } //創建一個Bean定義屬性值解析器,將Bean定義中的屬性值解析為Bean實例對象 //的實際值 BeanDefinitionValueResolver valueResolver = new BeanDefinitionValueResolver(this, beanName, mbd, converter); //為屬性的解析值創建一個拷貝,將拷貝的數據注入到實例對象中 List<PropertyValue> deepCopy = new ArrayList<PropertyValue>(original.size()); boolean resolveNecessary = false; for (PropertyValue pv : original) { //屬性值不需要轉換 if (pv.isConverted()) { deepCopy.add(pv); } //屬性值需要轉換 else { String propertyName = pv.getName(); //原始的屬性值,即轉換之前的屬性值 Object originalValue = pv.getValue(); //轉換屬性值,例如將引用轉換為IoC容器中實例化對象引用 Object resolvedValue = valueResolver.resolveValueIfNecessary(pv, originalValue); //轉換之后的屬性值 Object convertedValue = resolvedValue; //屬性值是否可以轉換 boolean convertible = bw.isWritableProperty(propertyName) && !PropertyAccessorUtils.isNestedOrIndexedProperty(propertyName); if (convertible) { //使用用戶自定義的類型轉換器轉換屬性值 convertedValue = convertForProperty(resolvedValue, propertyName, bw, converter); } //存儲轉換后的屬性值,避免每次屬性注入時的轉換工作 if (resolvedValue == originalValue) { if (convertible) { //設置屬性轉換之后的值 pv.setConvertedValue(convertedValue); } deepCopy.add(pv); } //屬性是可轉換的,且屬性原始值是字符串類型,且屬性的原始類型值不是 //動態生成的字符串,且屬性的原始值不是集合或者數組類型 else if (convertible && originalValue instanceof TypedStringValue && !((TypedStringValue) originalValue).isDynamic() && !(convertedValue instanceof Collection || ObjectUtils.isArray(convertedValue))) { pv.setConvertedValue(convertedValue); deepCopy.add(pv); } else { resolveNecessary = true; //重新封裝屬性的值 deepCopy.add(new PropertyValue(pv, convertedValue)); } } } if (mpvs != null && !resolveNecessary) { //標記屬性值已經轉換過 mpvs.setConverted(); } //進行屬性依賴注入 try { bw.setPropertyValues(new MutablePropertyValues(deepCopy)); } catch (BeansException ex) { throw new BeanCreationException( mbd.getResourceDescription(), beanName, "Error setting property values", ex); } }

    分析上述代碼,我們可以看出,對屬性的注入過程分以下兩種情況:

    (1).屬性值類型不需要轉換時,不需要解析屬性值,直接準備進行依賴注入。

    (2).屬性值需要進行類型轉換時,如對其他對象的引用等,首先需要解析屬性值,然后對解析后的屬性值進行依賴注入。

    對屬性值的解析是在BeanDefinitionValueResolver類中的resolveValueIfNecessary方法中進行的,對屬性值的依賴注入是通過bw.setPropertyValues方法實現的,在分析屬性值的依賴注入之前,我們先分析一下對屬性值的解析過程。

  • BeanDefinitionValueResolver解析屬性值:
  • 當容器在對屬性進行依賴注入時,如果發現屬性值需要進行類型轉換,如屬性值是容器中另一個Bean實例對象的引用,則容器首先需要根據屬性值解析出所引用的對象,然后才能將該引用對象注入到目標實例對象的屬性上去,對屬性進行解析的由resolveValueIfNecessary方法實現,其源碼如下:

    //解析屬性值,對注入類型進行轉換 public Object resolveValueIfNecessary(Object argName, Object value) { //對引用類型的屬性進行解析 if (value instanceof RuntimeBeanReference) { RuntimeBeanReference ref = (RuntimeBeanReference) value; //調用引用類型屬性的解析方法 return resolveReference(argName, ref); } //對屬性值是引用容器中另一個Bean名稱的解析 else if (value instanceof RuntimeBeanNameReference) { String refName = ((RuntimeBeanNameReference) value).getBeanName(); refName = String.valueOf(evaluate(refName)); //從容器中獲取指定名稱的Bean if (!this.beanFactory.containsBean(refName)) { throw new BeanDefinitionStoreException( "Invalid bean name '" + refName + "' in bean reference for " + argName); } return refName; } //對Bean類型屬性的解析,主要是Bean中的內部類 else if (value instanceof BeanDefinitionHolder) { BeanDefinitionHolder bdHolder = (BeanDefinitionHolder) value; return resolveInnerBean(argName, bdHolder.getBeanName(), bdHolder.getBeanDefinition()); } else if (value instanceof BeanDefinition) { BeanDefinition bd = (BeanDefinition) value; return resolveInnerBean(argName, "(inner bean)", bd); } //對集合數組類型的屬性解析 else if (value instanceof ManagedArray) { ManagedArray array = (ManagedArray) value; //獲取數組的類型 Class elementType = array.resolvedElementType; if (elementType == null) { //獲取數組元素的類型 String elementTypeName = array.getElementTypeName(); if (StringUtils.hasText(elementTypeName)) { try { //使用反射機制創建指定類型的對象 elementType = ClassUtils.forName(elementTypeName, this.beanFactory.getBeanClassLoader()); array.resolvedElementType = elementType; } catch (Throwable ex) { throw new BeanCreationException( this.beanDefinition.getResourceDescription(), this.beanName, "Error resolving array type for " + argName, ex); } } //沒有獲取到數組的類型,也沒有獲取到數組元素的類型,則直接設置數 //組的類型為Object else { elementType = Object.class; } } //創建指定類型的數組 return resolveManagedArray(argName, (List<?>) value, elementType); } //解析list類型的屬性值 else if (value instanceof ManagedList) { return resolveManagedList(argName, (List<?>) value); } //解析set類型的屬性值 else if (value instanceof ManagedSet) { return resolveManagedSet(argName, (Set<?>) value); } //解析map類型的屬性值 else if (value instanceof ManagedMap) { return resolveManagedMap(argName, (Map<?, ?>) value); } //解析props類型的屬性值,props其實就是key和value均為字符串的map else if (value instanceof ManagedProperties) { Properties original = (Properties) value; //創建一個拷貝,用于作為解析后的返回值 Properties copy = new Properties(); for (Map.Entry propEntry : original.entrySet()) { Object propKey = propEntry.getKey(); Object propValue = propEntry.getValue(); if (propKey instanceof TypedStringValue) { propKey = evaluate((TypedStringValue) propKey); } if (propValue instanceof TypedStringValue) { propValue = evaluate((TypedStringValue) propValue); } copy.put(propKey, propValue); } return copy; } //解析字符串類型的屬性值 else if (value instanceof TypedStringValue) { TypedStringValue typedStringValue = (TypedStringValue) value; Object valueObject = evaluate(typedStringValue); try { //獲取屬性的目標類型 Class<?> resolvedTargetType = resolveTargetType(typedStringValue); if (resolvedTargetType != null) { //對目標類型的屬性進行解析,遞歸調用 return this.typeConverter.convertIfNecessary(valueObject, resolvedTargetType); } //沒有獲取到屬性的目標對象,則按Object類型返回 else { return valueObject; } } catch (Throwable ex) { throw new BeanCreationException( this.beanDefinition.getResourceDescription(), this.beanName, "Error converting typed String value for " + argName, ex); } } else { return evaluate(value); } } //解析引用類型的屬性值 private Object resolveReference(Object argName, RuntimeBeanReference ref) { try { //獲取引用的Bean名稱 String refName = ref.getBeanName(); refName = String.valueOf(evaluate(refName)); //如果引用的對象在父類容器中,則從父類容器中獲取指定的引用對象 if (ref.isToParent()) { if (this.beanFactory.getParentBeanFactory() == null) { throw new BeanCreationException( this.beanDefinition.getResourceDescription(), this.beanName, "Can't resolve reference to bean '" + refName + "' in parent factory: no parent factory available"); } return this.beanFactory.getParentBeanFactory().getBean(refName); } //從當前的容器中獲取指定的引用Bean對象,如果指定的Bean沒有被實例化 //則會遞歸觸發引用Bean的初始化和依賴注入 else { Object bean = this.beanFactory.getBean(refName); //將當前實例化對象的依賴引用對象 this.beanFactory.registerDependentBean(refName, this.beanName); return bean; } } catch (BeansException ex) { throw new BeanCreationException( this.beanDefinition.getResourceDescription(), this.beanName, "Cannot resolve reference to bean '" + ref.getBeanName() + "' while setting " + argName, ex); } } //解析array類型的屬性 private Object resolveManagedArray(Object argName, List<?> ml, Class elementType) { //創建一個指定類型的數組,用于存放和返回解析后的數組 Object resolved = Array.newInstance(elementType, ml.size()); for (int i = 0; i < ml.size(); i++) { //遞歸解析array的每一個元素,并將解析后的值設置到resolved數組中,索引為i Array.set(resolved, i, resolveValueIfNecessary(new KeyedArgName(argName, i), ml.get(i))); } return resolved; } //解析list類型的屬性 private List resolveManagedList(Object argName, List<?> ml) { List<Object> resolved = new ArrayList<Object>(ml.size()); for (int i = 0; i < ml.size(); i++) { //遞歸解析list的每一個元素 resolved.add( resolveValueIfNecessary(new KeyedArgName(argName, i), ml.get(i))); } return resolved; } //解析set類型的屬性 private Set resolveManagedSet(Object argName, Set<?> ms) { Set<Object> resolved = new LinkedHashSet<Object>(ms.size()); int i = 0; //遞歸解析set的每一個元素 for (Object m : ms) { resolved.add(resolveValueIfNecessary(new KeyedArgName(argName, i), m)); i++; } return resolved; } //解析map類型的屬性 private Map resolveManagedMap(Object argName, Map<?, ?> mm) { Map<Object, Object> resolved = new LinkedHashMap<Object, Object>(mm.size()); //遞歸解析map中每一個元素的key和value for (Map.Entry entry : mm.entrySet()) { Object resolvedKey = resolveValueIfNecessary(argName, entry.getKey()); Object resolvedValue = resolveValueIfNecessary( new KeyedArgName(argName, entry.getKey()), entry.getValue()); resolved.put(resolvedKey, resolvedValue); } return resolved; }

    通過上面的代碼分析,我們明白了Spring是如何將引用類型,內部類以及集合類型等屬性進行解析的,屬性值解析完成后就可以進行依賴注入了,依賴注入的過程就是Bean對象實例設置到它所依賴的Bean對象屬性上去,在第7步中我們已經說過,依賴注入是通過bw.setPropertyValues方法實現的,該方法也使用了委托模式,在BeanWrapper接口中至少定義了方法聲明,依賴注入的具體實現交由其實現類BeanWrapperImpl來完成,下面我們就分析依BeanWrapperImpl中賴注入相關的源碼。

  • BeanWrapperImpl對Bean屬性的依賴注入:
  • BeanWrapperImpl類主要是對容器中完成初始化的Bean實例對象進行屬性的依賴注入,即把Bean對象設置到它所依賴的另一個Bean的屬性中去,依賴注入的相關源碼如下:

    //實現屬性依賴注入功能 private void setPropertyValue(PropertyTokenHolder tokens, PropertyValue pv) throws BeansException { //PropertyTokenHolder主要保存屬性的名稱、路徑,以及集合的size等信息 String propertyName = tokens.canonicalName; String actualName = tokens.actualName; //keys是用來保存集合類型屬性的size if (tokens.keys != null) { //將屬性信息拷貝 PropertyTokenHolder getterTokens = new PropertyTokenHolder(); getterTokens.canonicalName = tokens.canonicalName; getterTokens.actualName = tokens.actualName; getterTokens.keys = new String[tokens.keys.length - 1]; System.arraycopy(tokens.keys, 0, getterTokens.keys, 0, tokens.keys.length - 1); Object propValue; try { //獲取屬性值,該方法內部使用JDK的內省( Introspector)機制,調用屬性//的getter(readerMethod)方法,獲取屬性的值 propValue = getPropertyValue(getterTokens); } catch (NotReadablePropertyException ex) { throw new NotWritablePropertyException(getRootClass(), this.nestedPath + propertyName, "Cannot access indexed value in property referenced " + "in indexed property path '" + propertyName + "'", ex); } //獲取集合類型屬性的長度 String key = tokens.keys[tokens.keys.length - 1]; if (propValue == null) { throw new NullValueInNestedPathException(getRootClass(), this.nestedPath + propertyName, "Cannot access indexed value in property referenced " + "in indexed property path '" + propertyName + "': returned null"); } //注入array類型的屬性值 else if (propValue.getClass().isArray()) { //獲取屬性的描述符 PropertyDescriptor pd = getCachedIntrospectionResults().getPropertyDescriptor(actualName); //獲取數組的類型 Class requiredType = propValue.getClass().getComponentType(); //獲取數組的長度 int arrayIndex = Integer.parseInt(key); Object oldValue = null; try { //獲取數組以前初始化的值 if (isExtractOldValueForEditor()) { oldValue = Array.get(propValue, arrayIndex); } //將屬性的值賦值給數組中的元素 Object convertedValue = convertIfNecessary(propertyName, oldValue, pv.getValue(), requiredType, new PropertyTypeDescriptor(pd, new MethodParameter(pd.getReadMethod(), -1), requiredType)); Array.set(propValue, arrayIndex, convertedValue); } catch (IndexOutOfBoundsException ex) { throw new InvalidPropertyException(getRootClass(), this.nestedPath + propertyName, "Invalid array index in property path '" + propertyName + "'", ex); } } //注入list類型的屬性值 else if (propValue instanceof List) { PropertyDescriptor pd = getCachedIntrospectionResults().getPropertyDescriptor(actualName); //獲取list集合的類型 Class requiredType = GenericCollectionTypeResolver.getCollectionReturnType( pd.getReadMethod(), tokens.keys.length); List list = (List) propValue; //獲取list集合的size int index = Integer.parseInt(key); Object oldValue = null; if (isExtractOldValueForEditor() && index < list.size()) { oldValue = list.get(index); } //獲取list解析后的屬性值 Object convertedValue = convertIfNecessary(propertyName, oldValue, pv.getValue(), requiredType, new PropertyTypeDescriptor(pd, new MethodParameter(pd.getReadMethod(), -1), requiredType)); if (index < list.size()) { //為list屬性賦值 list.set(index, convertedValue); } //如果list的長度大于屬性值的長度,則多余的元素賦值為null else if (index >= list.size()) { for (int i = list.size(); i < index; i++) { try { list.add(null); } catch (NullPointerException ex) { throw new InvalidPropertyException(getRootClass(), this.nestedPath + propertyName, "Cannot set element with index " + index + " in List of size " + list.size() + ", accessed using property path '" + propertyName + "': List does not support filling up gaps with null elements"); } } list.add(convertedValue); } } //注入map類型的屬性值 else if (propValue instanceof Map) { PropertyDescriptor pd = getCachedIntrospectionResults().getPropertyDescriptor(actualName); //獲取map集合key的類型 Class mapKeyType = GenericCollectionTypeResolver.getMapKeyReturnType( pd.getReadMethod(), tokens.keys.length); //獲取map集合value的類型 Class mapValueType = GenericCollectionTypeResolver.getMapValueReturnType( pd.getReadMethod(), tokens.keys.length); Map map = (Map) propValue; //解析map類型屬性key值 Object convertedMapKey = convertIfNecessary(null, null, key, mapKeyType, new PropertyTypeDescriptor(pd, new MethodParameter(pd.getReadMethod(), -1), mapKeyType)); Object oldValue = null; if (isExtractOldValueForEditor()) { oldValue = map.get(convertedMapKey); } //解析map類型屬性value值 Object convertedMapValue = convertIfNecessary( propertyName, oldValue, pv.getValue(), mapValueType, new TypeDescriptor(new MethodParameter(pd.getReadMethod(), -1, tokens.keys.length + 1))); //將解析后的key和value值賦值給map集合屬性 map.put(convertedMapKey, convertedMapValue); } else { throw new InvalidPropertyException(getRootClass(), this.nestedPath + propertyName, "Property referenced in indexed property path '" + propertyName + "' is neither an array nor a List nor a Map; returned value was [" + pv.getValue() + "]"); } } //對非集合類型的屬性注入 else { PropertyDescriptor pd = pv.resolvedDescriptor; if (pd == null || !pd.getWriteMethod().getDeclaringClass().isInstance(this.object)) { pd = getCachedIntrospectionResults().getPropertyDescriptor(actualName); //無法獲取到屬性名或者屬性沒有提供setter(寫方法)方法 if (pd == null || pd.getWriteMethod() == null) { //如果屬性值是可選的,即不是必須的,則忽略該屬性值 if (pv.isOptional()) { logger.debug("Ignoring optional value for property '" + actualName + "' - property not found on bean class [" + getRootClass().getName() + "]"); return; } //如果屬性值是必須的,則拋出無法給屬性賦值,因為每天提供setter方法異常 else { PropertyMatches matches = PropertyMatches.forProperty(propertyName, getRootClass()); throw new NotWritablePropertyException( getRootClass(), this.nestedPath + propertyName, matches.buildErrorMessage(), matches.getPossibleMatches()); } } pv.getOriginalPropertyValue().resolvedDescriptor = pd; } Object oldValue = null; try { Object originalValue = pv.getValue(); Object valueToApply = originalValue; if (!Boolean.FALSE.equals(pv.conversionNecessary)) { if (pv.isConverted()) { valueToApply = pv.getConvertedValue(); } else { if (isExtractOldValueForEditor() && pd.getReadMethod() != null) { //獲取屬性的getter方法(讀方法),JDK內省機制 final Method readMethod = pd.getReadMethod(); //如果屬性的getter方法不是public訪問控制權限的,即訪問控制權限比較嚴格, //則使用JDK的反射機制強行訪問非public的方法(暴力讀取屬性值) if (!Modifier.isPublic(readMethod.getDeclaringClass().getModifiers()) && !readMethod.isAccessible()) { if (System.getSecurityManager()!= null) { //匿名內部類,根據權限修改屬性的讀取控制限制 AccessController.doPrivileged(new PrivilegedAction<Object>() { public Object run() { readMethod.setAccessible(true); return null; } }); } else { readMethod.setAccessible(true); } } try { //屬性沒有提供getter方法時,調用潛在的讀取屬性值//的方法,獲取屬性值 if (System.getSecurityManager() != null) { oldValue = AccessController.doPrivileged(new PrivilegedExceptionAction<Object>() { public Object run() throws Exception { return readMethod.invoke(object); } }, acc); } else { oldValue = readMethod.invoke(object); } } catch (Exception ex) { if (ex instanceof PrivilegedActionException) { ex = ((PrivilegedActionException) ex).getException(); } if (logger.isDebugEnabled()) { logger.debug("Could not read previous value of property '" + this.nestedPath + propertyName + "'", ex); } } } //設置屬性的注入值 valueToApply = convertForProperty(propertyName, oldValue, originalValue, pd); } pv.getOriginalPropertyValue().conversionNecessary = (valueToApply != originalValue); } //根據JDK的內省機制,獲取屬性的setter(寫方法)方法 final Method writeMethod = (pd instanceof GenericTypeAwarePropertyDescriptor ? ((GenericTypeAwarePropertyDescriptor) pd).getWriteMethodForActualAccess() : pd.getWriteMethod()); //如果屬性的setter方法是非public,即訪問控制權限比較嚴格,則使用JDK的反射機制, //強行設置setter方法可訪問(暴力為屬性賦值) if (!Modifier.isPublic(writeMethod.getDeclaringClass().getModifiers()) && !writeMethod.isAccessible()) { //如果使用了JDK的安全機制,則需要權限驗證 if (System.getSecurityManager()!= null) { AccessController.doPrivileged(new PrivilegedAction<Object>() { public Object run() { writeMethod.setAccessible(true); return null; } }); } else { writeMethod.setAccessible(true); } } final Object value = valueToApply; if (System.getSecurityManager() != null) { try { //將屬性值設置到屬性上去 AccessController.doPrivileged(new PrivilegedExceptionAction<Object>() { public Object run() throws Exception { writeMethod.invoke(object, value); return null; } }, acc); } catch (PrivilegedActionException ex) { throw ex.getException(); } } else { writeMethod.invoke(this.object, value); } } catch (TypeMismatchException ex) { throw ex; } catch (InvocationTargetException ex) { PropertyChangeEvent propertyChangeEvent = new PropertyChangeEvent(this.rootObject, this.nestedPath + propertyName, oldValue, pv.getValue()); if (ex.getTargetException() instanceof ClassCastException) { throw new TypeMismatchException(propertyChangeEvent, pd.getPropertyType(), ex.getTargetException()); } else { throw new MethodInvocationException(propertyChangeEvent, ex.getTargetException()); } } catch (Exception ex) { PropertyChangeEvent pce = new PropertyChangeEvent(this.rootObject, this.nestedPath + propertyName, oldValue, pv.getValue()); throw new MethodInvocationException(pce, ex); } } }

    通過對上面注入依賴代碼的分析,我們已經明白了Spring IoC容器是如何將屬性的值注入到Bean實例對象中去的:

    (1) 對于集合類型的屬性,將其屬性值解析為目標類型的集合后直接賦值給屬性。

    (2) 對于非集合類型的屬性,大量使用了JDK的反射和內省機制,通過屬性的getter方法(reader method)獲取指定屬性注入以前的值,同時調用屬性的setter方法(writer method)為屬性設置注入后的值。看到這里相信很多人都明白了Spring的setter注入原理。

    至此Spring IoC容器對Bean定義資源文件的定位,載入、解析和依賴注入已經全部分析完畢,現在Spring IoC容器中管理了一系列靠依賴關系聯系起來的Bean,程序不需要應用自己手動創建所需的對象,Spring IoC容器會在我們使用的時候自動為我們創建,并且為我們注入好相關的依賴,這就是Spring核心功能的控制反轉和依賴注入的相關功能。

    容器其他相關特性的設計與實現

    Application中Bean的初始化及銷毀

    簡要的介紹一下IoC容器中的Bean生命周期:

    • Bean實例的創建
    • 為Bean實例設置屬性
    • 調用Bean的初始化方法
    • 應用可以通過IoC容器使用Bean
    • 當容器關閉時,調用Bean的銷毀方法

    具體來說:

  • 實例化一個Bean--也就是我們常說的new;

  • 按照Spring上下文對實例化的Bean進行配置--也就是IOC注入;

  • 如果這個Bean已經實現了BeanNameAware接口,會調用它實現的setBeanName(String)方法,此處傳遞的就是Spring配置文件中Bean的id值;

  • 如果這個Bean已經實現了BeanFactoryAware接口,會調用它實現的setBeanFactory(setBeanFactory(BeanFactory)傳遞的是Spring工廠自身(可以用這個方式來獲取其它Bean,只需在Spring配置文件中配置一個普通的Bean就可以);

  • 如果這個Bean已經實現了ApplicationContextAware接口,會調用setApplicationContext(ApplicationContext)方法,傳入Spring上下文(同樣這個方式也可以實現步驟4的內容,但比4更好,因為ApplicationContext是BeanFactory的子接口,有更多的實現方法);

  • 如果這個Bean關聯了BeanPostProcessor接口,將會調用postProcessBeforeInitialization(Object obj, String s)方法,BeanPostProcessor經常被用作是Bean內容的更改,并且由于這個是在Bean初始化結束時調用那個的方法,也可以被應用于內存或緩存技術;

  • 如果bean實現了InitializingBean接口,spring將調用它的afterPropertiesSet接口方法;如果Bean在Spring配置文件中配置了init-method屬性會自動調用其配置的初始化方法。

  • 如果這個Bean關聯了BeanPostProcessor接口,將會調用postProcessAfterInitialization(Object obj, String s)方法、;
    注:以上工作完成以后就可以應用這個Bean了,那這個Bean是一個Singleton的,所以一般情況下我們調用同一個id的Bean會是在內容地址相同的實例,當然在Spring配置文件中也可以配置非Singleton,這里我們不做贅述。

  • 當Bean不再需要時,會經過清理階段,如果Bean實現了DisposableBean這個接口,會調用那個其實現的destroy()方法;

  • 最后,如果這個Bean的Spring配置中配置了destroy-method屬性,會自動調用其配置的銷毀方法。

  • 注:以上10步驟可以作為面試或者筆試的模板,另外我們這里描述的是應用Spring上下文Bean的生命周期,如果應用Spring的工廠也就是BeanFactory的話去掉第5步就Ok了。

    lazy-init屬性和預實例化

    IoC容器的初始化過程就是對Bean定義資源的定位、載入和注冊,此時容器對Bean的依賴注入并沒有發生,依賴注入主要是在應用程序第一次向容器索取Bean時,通過getBean方法的調用完成。 當Bean定義資源的< Bean >元素中配置了lazy-init屬性時,容器將會在初始化的時候對所配置的Bean進行預實例化,Bean的依賴注入在容器初始化的時候就已經完成。這樣,當應用程序第一次向容器索取被管理的Bean時,就不用再初始化和對Bean進行依賴注入了,直接從容器中獲取已經完成依賴注入的現成Bean,可以提高應用第一次向容器獲取Bean的性能。 當Bean定義資源被載入IoC容器之后,容器將Bean定義資源解析為容器內部的數據結構BeanDefinition注冊到容器中,AbstractApplicationContext類中的finishBeanFactoryInitialization方法對配置了預實例化屬性的Bean進行預初始化過程。

    FactoryBean的實現

    FactoryBean是一個典型的工廠模式的使用。它是一個Bean,作用是產生其他bean實例。通常情況下,僅需要提供一個工廠方法,該方法用來返回其他Bean實例。通常情況下,bean無須自己實現工廠模式,Spring容器擔任工廠角色;但少數情況下,容器中的bean本身就是工廠,其作用是產生其它Bean實例。

    當用戶使用容器本身時,可以使用轉義字符”&”來得到FactoryBean本身,以區別通過FactoryBean產生的實例對象和FactoryBean對象本身。在BeanFactory中通過如下代碼定義了該轉義字符。

    BeanPostProcessor的實現

    BeanPostProcessor后置處理器是Spring IoC容器經常使用到的一個特性,這個Bean后置處理器是一個監聽器,可以監聽容器觸發的Bean聲明周期事件。后置處理器向容器注冊以后,容器中管理的Bean就具備了接收IoC容器事件回調的能力。 BeanPostProcessor的使用非常簡單,只需要提供一個實現接口BeanPostProcessor的實現類,然后在Bean的配置文件中設置即可。

    package org.springframework.beans.factory.config; import org.springframework.beans.BeansException; public interface BeanPostProcessor { //為在Bean的初始化前提供回調入口 Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException; //為在Bean的初始化之后提供回調入口 Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException; }

    這兩個回調的入口都是和容器管理的Bean的生命周期事件緊密相關,可以為用戶提供在Spring IoC容器初始化Bean過程中自定義的處理操作。

    autowiring(自動裝配的實現)

    在自動裝配中,Spring IoC容器的依賴自動裝配功能,不需要對Bean屬性的依賴關系做顯式的聲明,只需要在配置好autowiring屬性,IoC容器會自動使用反射查找屬性的類型和名稱,然后基于屬性的類型或者名稱來自動匹配容器中管理的Bean,從而自動地完成依賴注入。

    應用第一次通過getBean方法(配置了lazy-init預實例化屬性的除外)向IoC容器索取Bean時,容器創建Bean實例對象,并且對Bean實例對象進行屬性依賴注入,AbstractAutoWireCapableBeanFactory的populateBean方法就是實現Bean屬性依賴注入的功能。

    在populateBean的實現中,在處理一般的Bean之前,先對autowiring屬性進行處理。如果當前的Bean配置了autowire_by_name和autowire_by_type屬性,那么調用相應的autowireByName方法autowireByType方法。對于autowire_by_name,它首先通過反射機制從當前Bean中得到需要注入的屬性名,然后使用這個屬性名向容器申請與之同名的Bean,這樣實際又觸發了另一個Bean的生成和依賴注入的過程。

    autowiring的實現過程:

    a. 對Bean的屬性迭代調用getBean方法,完成依賴Bean的初始化和依賴注入。

    b. 將依賴Bean的屬性引用設置到被依賴的Bean屬性上。

    c. 將依賴Bean的名稱和被依賴的Bean的名稱存儲在IoC容器的集合中。

    Bean的依賴檢查

    通過依賴檢查特性,Spring可以幫助應用檢查是否所有的屬性已經被正確設置。在具體使用的時候,應用只需要在Bean定義中設置dependency-check屬性來指定依賴檢查模式即可。
    這里可以將屬性設置為none、simple、object、all四種模式,默認的模式是none。

    總結

    以上是生活随笔為你收集整理的Spring IoC容器设计原理及高级特性的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    在线视频观看亚洲 | 国产精品二区在线观看 | 午夜影院先 | 亚洲另类xxxx | 91在线一区二区 | 国产在线自 | 亚洲精品乱码久久久久久久久久 | 精品国产免费一区二区三区五区 | 国产老太婆免费交性大片 | 亚洲精品午夜aaa久久久 | 日本精品久久久久影院 | 天天干天天看 | 亚洲黄色免费在线看 | 亚洲蜜桃在线 | 激情综合网天天干 | 99精品视频网站 | 精品国产1区| 日韩精品电影在线播放 | 又污又黄网站 | 欧美激情精品 | 丁香六月伊人 | 国产精品资源在线 | 91精品在线看 | 亚洲视频免费在线看 | 亚洲片在线资源 | 黄色免费网 | 国产精品久久嫩一区二区免费 | www·22com天天操 | 久久国产精品99久久人人澡 | 天天操综合网站 | 国产区欧美 | 91高清完整版在线观看 | 波多野结衣精品在线 | 欧洲亚洲国产视频 | 久久国产精品色av免费看 | 久久久精品 一区二区三区 国产99视频在线观看 | 久久激情综合网 | 在线亚洲成人 | 在线中文字幕电影 | 成人小视频免费在线观看 | 亚洲激情小视频 | 久久精品视频播放 | 久久一二区 | 超碰最新网址 | 麻豆影视网站 | 国产一级做a | 婷婷电影在线观看 | 精品国产一二三 | 中文字幕专区高清在线观看 | 亚洲视频一区二区三区在线观看 | 国产高清视频网 | 久久情爱 | 精品成人在线 | 综合影视| 久久国产精品成人免费浪潮 | 91精品对白一区国产伦 | 97精品国产91久久久久久久 | 国产欧美日韩一区 | 亚洲欧美成人综合 | 可以免费看av | 天天色天天操综合 | 国产精品入口久久 | 免费观看视频黄 | 黄av在线 | 国产婷婷久久 | 国产一级特黄电影 | 青青河边草观看完整版高清 | 久久午夜色播影院免费高清 | 天天综合网久久综合网 | 亚洲人成在线电影 | 国产电影黄色av | 欧美狠狠操 | 国产成免费视频 | 中文字幕xxxx | 97精品国产一二三产区 | 黄色av三级在线 | 国内精品在线观看视频 | 色多多污污 | 一色屋精品视频在线观看 | 黄污视频网站大全 | 黄色av免费电影 | 午夜av免费 | 亚洲影院天堂 | 日韩肉感妇bbwbbwbbw | 婷婷色网 | 青青草国产成人99久久 | 国产美女视频网站 | 中文字幕a在线 | 国产99精品 | 亚洲国内精品在线 | 午夜精品久久久久久 | 97免费视频在线 | 超碰97网站 | 香蕉视频在线播放 | 婷婷久草| 97精品国产97久久久久久粉红 | 96精品高清视频在线观看软件特色 | 国产原厂视频在线观看 | 久久久久影视 | 国产精品大片在线观看 | 日韩va欧美va亚洲va久久 | 一区二区三区免费播放 | 国产视频久久 | 国产亚洲精品日韩在线tv黄 | 高清国产午夜精品久久久久久 | 亚洲免费成人 | 五月婷av | 欧美一区二区精品在线 | 午夜 免费| 99久久精品免费看 | 久久美女高清视频 | 国产清纯在线 | 久久视频在线观看 | 在线视频专区 | 精品久久久一区二区 | 四虎在线观看精品视频 | 国产69精品久久app免费版 | 丁香六月婷婷开心 | 精品一区二区三区香蕉蜜桃 | 中文字幕在线观看免费观看 | 激情五月亚洲 | 超级碰碰视频 | 精品福利视频在线观看 | 在线成人免费电影 | 日韩欧美在线综合网 | 亚洲精品国偷自产在线99热 | 国产成人一区二区三区影院在线 | 久久这里只有精品视频首页 | 免费看黄色91 | 亚洲最新av | 国产成人精品一区二区三区福利 | 女人18片毛片90分钟 | 成人精品福利 | 亚洲精品影院在线观看 | 中文av影院 | www.狠狠插.com | 欧美久久成人 | 91成年人在线观看 | 2019中文| 国产日本在线观看 | 国产精品美女久久久久久2018 | 中文字幕在线观看第一区 | 久久草在线视频国产 | 国产一区二区精品 | 日韩中文字幕第一页 | 成片视频在线观看 | 国精产品一二三线999 | 天天天干天天天操 | 视频在线播放国产 | 国产最新视频在线 | 四虎视频 | 午夜私人影院久久久久 | 黄色免费看片网站 | 天天操天天射天天操 | 久久99在线观看 | 97超碰国产精品 | 国产精品一区在线 | 国产精品一区二区在线播放 | 99精品视频中文字幕 | 一区二区三区影院 | 亚洲国产日韩欧美在线 | 西西www4444大胆视频 | 91一区二区三区在线观看 | 久久久男人的天堂 | 亚洲人成精品久久久久 | 制服丝袜亚洲 | 日韩一级精品 | 91污视频在线观看 | 日韩区欧美久久久无人区 | 亚洲男女精品 | 波多野结衣在线播放一区 | 亚州精品天堂中文字幕 | 九九九热视频 | 免费在线观看av | 蜜臀久久99精品久久久久久网站 | 国产精品一区二区久久久久 | 成人cosplay福利网站 | 国产精品自产拍在线观看 | 免费在线观看成人av | 黄色毛片观看 | 国产精品久久久久久久av电影 | 在线视频福利 | 成人免费电影 | 激情欧美国产 | 国产成人99久久亚洲综合精品 | 欧美一级片播放 | 一区二区精品久久 | av高清在线观看 | 免费看黄在线 | 久久论理 | 欧美乱熟臀69xxxxxx | 西西www444 | 国产免费xvideos视频入口 | 91插插插网站 | 69精品视频 | 亚洲午夜久久久久 | 日日夜夜免费精品 | 色婷婷国产 | 国产精品999久久久 久产久精国产品 | 91久久一区二区 | 久久99热这里只有精品 | 中文字幕91视频 | 黄色视屏在线免费观看 | 日韩一区二区三区免费视频 | 亚洲精品在线视频播放 | 欧美精品久久久久久久久老牛影院 | 在线看一区 | 99中文视频在线 | 少妇自拍av | 精品免费视频 | 色多视频在线观看 | 在线免费91| 97品白浆高清久久久久久 | 日韩欧美xxx | 免费观看成年人视频 | 亚洲国产欧美在线人成大黄瓜 | 亚洲综合最新在线 | 亚洲综合爱 | 国产精品日韩 | 高清国产在线一区 | 国精产品满18岁在线 | 性色av一区二区三区在线观看 | 欧美精品首页 | 美女黄色网在线播放 | av福利网址导航 | 狠狠色丁香婷婷综合最新地址 | 亚洲欧美日韩精品久久久 | 在线视频 91| 黄污网站在线 | 999国产精品视频 | 国产 在线观看 | 色黄www小说 | 亚洲视频 中文字幕 | 欧美精品二区 | 国产91全国探花系列在线播放 | 久久精品一区二区三 | 精品国产一区二区三区久久影院 | 日本h在线播放 | 91探花国产综合在线精品 | 又黄又刺激视频 | 97视频免费在线 | 在线观看视频国产一区 | 国产理论影院 | 免费观看性生活大片 | 亚洲一区二区三区精品在线观看 | 欧美色精品天天在线观看视频 | 天天射网| 国产精品原创在线 | zzijzzij亚洲日本少妇熟睡 | 亚洲综合色激情五月 | 999久久久免费精品国产 | 午夜av影院 | 亚洲国产福利视频 | 国产一区二区精品久久 | 91麻豆免费视频 | 黄色成人小视频 | 中国一 片免费观看 | 国产无遮挡猛进猛出免费软件 | 国产精品美女免费视频 | www.久久久.cum | 亚洲精品久久激情国产片 | 国产大尺度视频 | 国产美女被啪进深处喷白浆视频 | 中文字幕在线观看免费 | 亚洲视频 视频在线 | 91精品国产亚洲 | 精品国产乱码久久久久久浪潮 | av不卡在线看 | 欧美va在线观看 | 中文字幕乱视频 | 久插视频 | 天天做天天射 | 欧美国产日韩在线观看 | 久久99精品国产麻豆宅宅 | 国产精品网址在线观看 | 日本一区二区不卡高清 | 91精品网站在线观看 | 天天操夜夜操夜夜操 | 日韩在线一区二区免费 | 精品96久久久久久中文字幕无 | 国产精品一区二区三区99 | 大荫蒂欧美视频另类xxxx | 在线观看国产区 | 欧美日韩三级 | 国产精品黑丝在线观看 | 国产专区欧美专区 | 国产一级电影网 | a资源在线 | 亚洲免费在线观看视频 | 免费av大片 | 久久久久久久久福利 | 久草视频在线资源站 | www.天天操 | 欧美一区日韩精品 | 九九精品久久 | 在线国产一区二区三区 | 久保带人 | 一性一交视频 | 开心综合网 | 精品在线播放 | 99精品在线免费视频 | 久久久蜜桃 | 九色porny真实丨国产18 | 亚洲精品乱码久久久久久蜜桃91 | 午夜av大片 | 狠狠操操 | 国内精品免费久久影院 | 五月天亚洲精品 | 日韩伦理片一区二区三区 | 99精品一级欧美片免费播放 | 久久九九国产视频 | 国产美女黄网站免费 | 成人av视屏 | 国产精品视频不卡 | 91热精品| 免费看黄的视频 | 久久成人国产 | 亚洲精品视频二区 | 精品免费在线视频 | 青青色影院 | 在线观看亚洲 | 亚洲丝袜中文 | 国产女人免费看a级丨片 | 久久99在线| 91精品国产92久久久久 | 在线v片 | 成人黄大片 | 99久久精品国产毛片 | 国产精品手机视频 | 在线中文字幕视频 | 91视频国产高清 | www.天天操.com | av免费在线观 | 国产精品欧美一区二区三区不卡 | 一二三区高清 | 国产一区二区三区黄 | 天天夜夜亚洲 | 国产麻豆视频免费观看 | 日本在线视频一区二区三区 | 日韩精品91偷拍在线观看 | 波多野结衣电影久久 | www.久久爱.cn | 狠狠操狠狠操 | 久久全国免费视频 | 蜜桃av久久久亚洲精品 | 成年人免费在线观看网站 | 久久久久女人精品毛片九一 | 99久久精品免费看国产一区二区三区 | 久久久久免费精品视频 | 国产美女在线精品免费观看 | 国内揄拍国产精品 | 久久ww| 在线欧美日韩 | 在线国产99| 国产69精品久久99不卡的观看体验 | 精品国产三级 | 97超碰国产精品 | 国产一区二区三区在线 | 欧美三人交 | 天天操天天操 | 奇米四色影狠狠爱7777 | 一区二区三区精品久久久 | 欧美精品一区二区在线播放 | 亚洲精品一区二区网址 | 亚洲电影av在线 | 手机看片国产日韩 | 500部大龄熟乱视频使用方法 | 国产专区在线视频 | 综合久久久 | 一区二区三区视频网站 | 97色免费视频 | 国产精品一二三 | 97日日 | 欧美日韩在线免费观看视频 | 国内精品国产三级国产aⅴ久 | 狠狠色伊人亚洲综合网站野外 | 日韩欧美一区二区在线 | 国产高清第一页 | 国产这里只有精品 | 亚洲精品永久免费视频 | 嫩模bbw搡bbbb搡bbbb | avsex| 免费色网 | 亚洲黄色av网址 | 91热这里只有精品 | 91成人天堂久久成人 | 在线看国产 | 欧美日韩不卡在线视频 | 国产无限资源在线观看 | 中文字幕日韩免费视频 | 亚洲精品在线免费观看视频 | 日免费视频 | 国产精品网站一区二区三区 | 天天操天天射天天舔 | 色中色资源站 | 色综合天天综合在线视频 | 91福利视频久久久久 | 中文字幕在线播放一区 | 亚洲免费观看在线视频 | 9ⅰ精品久久久久久久久中文字幕 | 在线日韩亚洲 | 国产精品久久久久久久电影 | 日韩免费电影在线观看 | 国产色视频一区二区三区qq号 | 国产手机在线 | 日本精品一 | 中文永久字幕 | 一区二区三区在线不卡 | 九色91视频 | 在线精品一区二区 | 美女久久久久久久 | 天天天干天天射天天天操 | 91毛片在线观看 | 国产18精品乱码免费看 | 亚洲视频免费 | 国产精品va最新国产精品视频 | 久久精彩视频 | 开心激情婷婷 | 久久这里只有精品1 | 欧美国产视频在线 | 黄色在线观看免费网站 | 岛国大片免费视频 | 99国产视频| 97小视频| 五月婷婷六月丁香 | 狠狠色噜噜狠狠狠合久 | 日韩av片无码一区二区不卡电影 | 国产色女 | 国产成人久久77777精品 | 亚洲国产三级在线 | 激情电影在线观看 | 国产视频不卡一区 | 国内精品久久久精品电影院 | 狠狠干免费 | 国产一区在线不卡 | 中文字幕 国产视频 | 日本激情视频中文字幕 | 五月婷色 | 青青看片 | 久久综合中文字幕 | 天天艹日日干 | 免费观看视频黄 | 国产精品永久久久久久久久久 | 精品久久久影院 | 1000部18岁以下禁看视频 | 久久精品第一页 | 日韩av区 | 亚洲一区美女视频在线观看免费 | 成年人免费av网站 | 99精品在线免费视频 | 久久在线一区 | 成人午夜剧场在线观看 | 爱色av.com| 亚洲欧美日韩国产 | 中文在线√天堂 | 一区二区三区www | 中文字幕电影网 | 久久香蕉电影网 | 中文在线8新资源库 | 色 免费观看 | 97av在线视频| 五月天丁香综合 | 国产精品理论视频 | 日韩视频免费在线观看 | 久草视频在 | 久久久久久久久久久高潮一区二区 | 国产日产亚洲精华av | 综合婷婷久久 | 日韩精品久久一区二区三区 | 国产一级片网站 | 狠狠亚洲 | 日韩在线观看一区二区三区 | www91在线观看 | 18岁免费看片 | 久久久久五月天 | 狠狠久久综合 | 久久99精品久久久久婷婷 | www.xxxx欧美| 亚洲欧洲一区二区在线观看 | a黄色一级 | 久久视频在线免费观看 | 久久久久99精品成人片三人毛片 | 丝袜美腿在线 | 国产成人福利片 | 亚洲免费在线观看视频 | 天天操天天弄 | 欧美日本中文字幕 | 国产一及片 | 天天天色 | 黄色网址中文字幕 | 亚洲高清视频在线 | 色停停五月天 | 91在线看 | 国产一二区视频 | 黄色精品网站 | 免费69视频| 精品久久久久久综合日本 | 色综合天天色综合 | www.久久久.com | 欧美成人手机版 | 日韩在线观看视频免费 | 一区二区三区四区在线 | 欧美视频不卡 | 中文字幕专区高清在线观看 | 美女在线免费视频 | 91手机视频 | 日本天天操 | 成人午夜av电影 | 国产999精品久久久影片官网 | 亚洲视频综合在线 | 91看片一区二区三区 | 伊人五月综合 | 91干干干 | 中文字幕婷婷 | 天天综合色 | 九九热精品在线 | 精品国产区 | 久久精品电影院 | 国产污视频在线观看 | 免费看片网页 | 精品一区精品二区 | 西西大胆免费视频 | 又黄又爽又色无遮挡免费 | 99精品在线播放 | 国产丝袜一区二区三区 | 免费观看一区二区 | 亚洲一区二区三区miaa149 | 精品国产1区2区3区 国产欧美精品在线观看 | 成人蜜桃 | 亚洲一区二区精品在线 | 在线免费观看国产精品 | 欧美一级免费黄色片 | av福利在线 | 日韩在线资源 | 日韩一区在线播放 | 免费观看一级一片 | 久久精品一区二区三区国产主播 | 久久久久久99精品 | 午夜精品av| 奇米先锋 | 亚洲精品国偷自产在线99热 | 国产一区二区观看 | 在线视频欧美精品 | 色婷婷国产精品一区在线观看 | 亚洲综合在线五月 | 久久精品永久免费 | 五月天婷婷视频 | 丁香激情综合久久伊人久久 | av福利在线免费观看 | 国产精品av在线 | 中文字幕在线观看2018 | 在线99热 | 精品久操 | 免费日韩视频 | 国产精彩视频一区二区 | 天天色天| 国产一区久久久 | 久久综合色婷婷 | 久久99国产精品久久 | 日韩午夜三级 | www.天天色| 国产精品99爱 | 免费a v在线| 国产成人l区 | 亚洲精品免费在线观看视频 | 久久久久99999 | 亚洲资源在线观看 | 亚洲美女免费精品视频在线观看 | 欧美精品一区在线 | 天天婷婷| 在线免费观看视频一区二区三区 | 91av中文| 久久在线视频精品 | 国产精品久久久久永久免费观看 | 六月激情久久 | 久草免费手机视频 | 在线免费av网 | 日韩电影精品一区 | 久久综合久久综合这里只有精品 | 欧美性生爱| 13日本xxxxxⅹxxx20 | 午夜精品久久久久久久99婷婷 | 日日成人网| 久青草影院 | 97久久久免费福利网址 | 欧美日韩亚洲国产一区 | 国产成人在线一区 | 日本xxxx.com | 亚洲女欲精品久久久久久久18 | 精品国产一区二区三区不卡 | 久久久久久久久久久久久影院 | 亚洲春色奇米影视 | 国产福利一区二区三区在线观看 | 久久久久99精品成人片三人毛片 | 国产一区av在线 | 日韩美一区二区三区 | 日韩欧美一区二区在线观看 | 亚洲免费成人av电影 | 在线亚洲播放 | 96精品在线 | 日韩精品一区二区三区不卡 | 九九亚洲精品 | 蜜臀久久99精品久久久久久网站 | 粉嫩av一区二区三区四区 | 日韩免费一区二区三区 | 91视频在线播放视频 | a级国产乱理论片在线观看 特级毛片在线观看 | 亚洲黄色一级电影 | 天天操天天射天天爱 | 日韩av一区二区在线影视 | 99精品视频中文字幕 | 欧美日韩精品在线一区二区 | 久久不卡免费视频 | 国产r级在线观看 | 国产视频日韩视频欧美视频 | 亚洲人天堂 | 日本久久视频 | 亚洲免费永久精品国产 | 久久成人综合视频 | 欧美日韩二区三区 | 99在线精品视频在线观看 | 亚州av一区 | 欧美日韩国产二区 | 成人午夜电影在线观看 | 蜜臀av性久久久久蜜臀aⅴ四虎 | 亚欧洲精品视频在线观看 | 国内视频在线观看 | 极品美女被弄高潮视频网站 | 久久免费视频6 | 中文字幕在线免费播放 | 婷婷免费在线视频 | 日本一区二区免费在线观看 | 区一区二区三区中文字幕 | 九九av | 国产精品视屏 | 西西www444 | 极品美女被弄高潮视频网站 | 国产精品久久久久影视 | 婷婷亚洲最大 | 九色福利视频 | 久久精品成人欧美大片古装 | 手机看片1042 | 成人夜晚看av | 国产97视频 | 99久久精品视频免费 | 日韩精品视频在线观看网址 | 欧美国产日韩一区二区三区 | 亚洲二级片 | 久久综合网色—综合色88 | 中文字幕在线色 | 一区二区三区免费 | 色综合色综合色综合 | 久久观看免费视频 | 激情导航| 免费看的毛片 | 久久精品成人欧美大片古装 | 玖玖在线视频观看 | 国产 日韩 在线 亚洲 字幕 中文 | 国精产品满18岁在线 | 日韩电影中文,亚洲精品乱码 | 国产精品久久久久国产a级 激情综合中文娱乐网 | 最新av在线免费观看 | 久久综合九色综合久99 | 国产一区视频在线观看免费 | 免费看一及片 | 97视频入口免费观看 | 国产伦理久久 | 精品视频免费观看 | 成人久久国产 | 国产伦理一区二区三区 | 日韩精品一区二区三区第95 | 日韩中文字幕免费 | 91片黄在线观看动漫 | 欧美尹人 | 国产精品色在线 | 国产精品精品久久久久久 | 狠狠操夜夜 | 97成人精品视频在线观看 | 五月激情婷婷丁香 | 操久在线| 一区二区三区在线观看免费视频 | 成年人免费观看国产 | 日韩在线不卡av | 狠狠狠狠狠狠操 | 国产日韩欧美网站 | 狠狠的操狠狠的干 | 久久精品一区二区三 | 国产免费av一区二区三区 | 美女网站在线播放 | 国产精品久久久久永久免费观看 | 2023亚洲精品国偷拍自产在线 | 91精品国产91久久久久福利 | 高清久久久久久 | 中文字幕第一页在线播放 | 久久久亚洲国产精品麻豆综合天堂 | 亚洲人在线视频 | 亚洲精品免费观看视频 | 九九九视频精品 | 日韩理论电影在线观看 | 一区二区三区在线免费观看视频 | 国内精品久久天天躁人人爽 | 色一级片 | 中文字幕av全部资源www中文字幕在线观看 | 婷婷六月在线 | 精品视频久久久久久 | 国产成人av网 | 亚洲黄色免费网站 | 在线观看日韩视频 | 国产麻豆精品一区二区 | 美女网站视频久久 | 久久网页 | 欧美日韩国产精品一区二区亚洲 | 亚洲综合色丁香婷婷六月图片 | 911久久 | 久久久av免费 | 免费在线观看av网址 | 麻豆精品在线视频 | 色综合天天综合 | 久久夜色精品亚洲噜噜国4 午夜视频在线观看欧美 | 在线观看网站你懂的 | 久久y| 亚洲小视频在线观看 | 探花在线观看 | 99精品国产免费久久久久久下载 | 天天操网站 | 欧美韩日视频 | 国产精品18久久久久久首页狼 | 中文国产在线观看 | 精品欧美一区二区三区久久久 | 91成人看片 | 久久99国产综合精品免费 | 99这里只有| 天天插狠狠插 | 久久久男人的天堂 | 伊人影院99| 日韩天堂网 | 欧洲一区二区在线观看 | 手机在线看a | 亚洲激情在线观看 | 国产美女精品在线 | 亚洲高清在线 | 午夜免费福利片 | 在线视频免费观看 | 国产伦精品一区二区三区四区视频 | 在线午夜电影神马影院 | 国产精品嫩草影视久久久 | 免费看的黄色小视频 | 久久久精品免费看 | 国产亚洲精品无 | 亚洲精品人人 | av色综合网 | 黄色av免费电影 | 超碰免费久久 | 国产在线观看二区 | 久久久国产精品电影 | 国内久久视频 | 国产999精品久久久影片官网 | 亚洲美女精品 | 精品在线视频一区二区三区 | 成人黄色在线 | 香蕉视频4aa | 欧美精品久久久久久久久久 | 亚洲精品免费播放 | 99久久99久久| 丁香婷婷深情五月亚洲 | 日韩v欧美v日本v亚洲v国产v | 国产成人1区 | 国产午夜三级一区二区三桃花影视 | 黄色三几片 | 国产二区视频在线观看 | 久久精品一区二区三区中文字幕 | 欧美日韩国产精品一区二区亚洲 | 最新日韩在线观看 | 国产精品成人在线 | 特及黄色片| 黄污视频网站大全 | 午夜av大片 | 久久综合欧美精品亚洲一区 | 久久久伊人网 | 不卡的av | 伊人久久精品久久亚洲一区 | 国产情侣一区 | 色视频在线免费观看 | 一区二区三区在线不卡 | 99国产一区二区三精品乱码 | 这里只有精品视频在线 | 91网址在线观看 | 日韩欧美国产免费播放 | 天天射天天干 | 91免费网址| 91免费版成人 | 免费观看全黄做爰大片国产 | 国内视频 | 久久国产免 | 二区视频在线观看 | 免费情趣视频 | 在线看片中文字幕 | 欧美精品亚州精品 | 成人黄色av免费在线观看 | 亚洲精品mv在线观看 | 欧美精彩视频在线观看 | 网址你懂的在线观看 | 婷婷色婷婷 | 夜夜干天天操 | 最新日韩在线观看视频 | 超碰公开在线观看 | 亚洲国产三级在线 | 国产成人精品一区在线 | 亚洲va欧洲va国产va不卡 | 91在线免费播放 | 99免费视频| 久久国产乱| 婷久久 | 天天综合网~永久入口 | 中文区中文字幕免费看 | 欧美日韩超碰 | 久久免费的视频 | 国产黄色在线网站 | 人人澡人 | 一区二区理论片 | 国产色婷婷| 免费福利在线观看 | 免费在线观看一区 | 欧美日韩视频观看 | a级片网站 | 中文区中文字幕免费看 | 免费视频成人 | 日韩黄色免费 | 免费在线观看午夜视频 | 日韩午夜网站 | 在线观看中文字幕亚洲 | 国产一级在线看 | 黄色大片入口 | 色网站视频 | 国产xvideos免费视频播放 | 久久精品国产第一区二区三区 | 蜜臀久久99精品久久久无需会员 | 在线小视频你懂的 | 国产一区二区三区视频在线 | 国产午夜精品一区 | 天天激情 | 在线看国产一区 | 中文字幕在线观看av | 国产精品自产拍在线观看网站 | 91传媒在线播放 | 99日韩精品 | 在线免费黄 | 在线视频黄| 久久精品小视频 | 国产精品免费视频一区二区 | 免费观看性生交 | 玖玖玖精品 | 免费看毛片网站 | 91在线观 | av中文天堂 | 天天天天综合 | 天天综合网天天综合色 | 天干啦夜天干天干在线线 | 成人一区二区在线观看 | 欧美精品三级在线观看 | 国产美女被啪进深处喷白浆视频 | 久久精品99国产国产精 | 天天干天天插伊人网 | 在线观看中文字幕dvd播放 | 亚洲黄色免费网站 | 欧美日韩在线视频观看 | 久要激情网 | 日本黄色免费网站 | 欧美大码xxxx| 天天插狠狠干 | 天天射色综合 | 天天综合入口 | 日韩欧美有码在线 | 国产视频一区精品 | 免费a视频在线观看 | 在线观看中文 | 欧美成年性 | 天天爱天天操天天干 | 色播激情五月 | 日本中文字幕在线看 | 999男人的天堂| 日韩一三区| 亚洲免费av片 | 视频在线国产 | 日韩大陆欧美高清视频区 | www.色国产 | 国产精品成人一区二区三区吃奶 | 手机av片| 国产理论免费 | 欧美日韩视频在线观看一区二区 | 亚洲91在线 | 久久黄视频| 久久精品网站视频 | 三级视频片 | 在线观看av麻豆 | 久久婷婷一区 | 欧美aa一级片 | 国产在线一卡 | 国产一二三四在线视频 | 成年人国产在线观看 | 日p视频在线观看 | 91人人揉日日捏人人看 | 91久久国产综合精品女同国语 | 国色天香在线 | 视频国产在线 | 欧美日韩在线观看不卡 | 热久精品| 亚洲精品视频一 | 九九影视理伦片 | 中文字幕av在线电影 | 国内精品久久久久影院一蜜桃 | 色婷久久 | 精品视频免费播放 | 久久久精品欧美一区二区免费 | 国产精品视频免费观看 | 婷婷五综合 | 三级av小说| 亚洲国产欧美在线人成大黄瓜 | 久久精品精品电影网 | 欧美99久久 | 婷婷激情五月 | 天天综合网~永久入口 | 色在线高清 | 亚洲九九| 精品久久免费看 | 国产精品a成v人在线播放 | 成人av在线播放网站 | 少妇搡bbbb搡bbb搡69 | 天天操天天草 | 欧美日本不卡视频 | 国产亚洲在线视频 | 国产一区二区在线免费播放 | 欧美一区二区在线刺激视频 | 日日草夜夜操 | 丁香视频五月 | 在线免费国产视频 | 国内三级在线 | 青青网视频| 国产资源免费 | 日韩欧美精品在线视频 | av不卡网站 | 久久都是精品 | av在观看 | 国产区精品区 | 一本色道久久精品 | 午夜视频一区二区三区 | 成年人免费电影在线观看 | 黄色免费网站 | 最新中文字幕在线观看视频 | 日日操天天操狠狠操 | 国产在线视频在线观看 | 992tv在线观看网站 | 国产精品亚洲视频 | 久久精品在线 | 在线免费观看视频 | 99在线精品视频在线观看 | 在线观看完整版免费 | 在线观看成人毛片 | 视频国产区 | 中文字幕亚洲国产 | 国产手机视频精品 | 久草在线资源免费 | 国产日韩欧美在线一区 | 久久综合五月天 | 国产福利精品在线观看 | 国产精品不卡视频 | 日韩av在线小说 | 又黄又爽又无遮挡的视频 | 国内精品视频免费 | 国产香蕉视频 | 欧美激情精品久久久久久免费印度 | 国内少妇自拍视频一区 | 激情欧美一区二区三区 | 国产淫a| 51久久夜色精品国产麻豆 | 亚洲国产成人久久 | 欧美九九视频 | 在线视频久久 | 欧美在线一级片 | 91大神一区二区三区 | 超碰人人av| 国产不卡视频在线播放 | 色综合欧洲 | 日韩黄色在线电影 | 欧美久久久久久久久 | 日日爽 | 日p视频 | 久久深夜 | 亚洲精品五月 | 国产亚洲精品久久久久久久久久久久 | 久久精品站 | av7777777 | 国产色区 | 免费视频黄 | 久久久91精品国产 | 国产亚洲精品v | 精品a视频 | 国产黄网在线 |