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

Spring源碼之Bean實例化基本原理

開發 前端
在實例化Bean之前在BeanDefinition里頭已經有了所有需要實例化時用到的元數據,接下來Spring只需要選擇合適的實例化方法以及策略即可。

 [[346085]]

創建Spring Bean實例化是Spring Bean生命周期的第一階段

Bean的生命周期主要有如下幾個步驟:

「詳細介紹:Spring In Action是這樣講的:」

  • 實例化Bean對象,這個時候Bean的對象是非常低級的,基本不能夠被我們使用,因為連最基本的屬性都沒有設置,可以理解為連Autowired注解都是沒有解析的;
  • 填充屬性,當做完這一步,Bean對象基本是完整的了,可以理解為Autowired注解已經解析完畢,依賴注入完成了;
  • 如果Bean實現了BeanNameAware接口,則調用setBeanName方法;
  • 如果Bean實現了BeanClassLoaderAware接口,則調用setBeanClassLoader方法;
  • 如果Bean實現了BeanFactoryAware接口,則調用setBeanFactory方法;
  • 調用BeanPostProcessor的postProcessBeforeInitialization方法;
  • 如果Bean實現了InitializingBean接口,調用afterPropertiesSet方法;
  • 如果Bean定義了init-method方法,則調用Bean的init-method方法;
  • 調用BeanPostProcessor的postProcessAfterInitialization方法;當進行到這一步,Bean已經被準備就緒了,一直停留在應用的上下文中,直到被銷毀;
  • 如果應用的上下文被銷毀了,如果Bean實現了DisposableBean接口,則調用destroy方法,如果Bean定義了destory-method聲明了銷毀方法也會被調用。

在實例化Bean之前在BeanDefinition里頭已經有了所有需要實例化時用到的元數據,接下來Spring只需要選擇合適的實例化方法以及策略即可。

「BeanDefinition」

Spring容器啟動的時候會定位我們的配置文件,加載文件,并解析成Bean的定義文件BeanDefinition

右邊的Map里存儲這bean之間的依賴關系的定義BeanDefinition,比如OrderController依賴OrderService這種

實例化方法有兩大類分別是工廠方法和構造方法實例化,后者是最常見的。其中Spring默認的實例化方法就是無參構造函數實例化。

如我們在xml里定義的以及用注解標識的bean都是通過默認實例化方法實例化的

實例化方法

「使靜態工廠方法實例化」

  1. public class FactoryInstance { 
  2.  
  3.     public FactoryInstance() { 
  4.         System.out.println("instance by FactoryInstance"); 
  5.     } 
  1. public class MyBeanFactory { 
  2.  
  3.     public static FactoryInstance getInstanceStatic(){ 
  4.         return new FactoryInstance(); 
  5.     } 
  1. <?xml version="1.0" encoding="UTF-8"?> 
  2. <beans xmlns="http://www.springframework.org/schema/beans" 
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
  4.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"
  5.  
  6.     <bean id="factoryInstance" class="spring.service.instance.MyBeanFactory"  
  7.           factory-method="getInstanceStatic"/> 
  8. </beans> 

「使用實例工廠方法實例化」

  1. public class MyBeanFactory { 
  2.  
  3.     /** 
  4.      * 實例工廠創建bean實例 
  5.      * 
  6.      * @return 
  7.      */ 
  8.     public FactoryInstance getInstance() { 
  9.         return new FactoryInstance(); 
  10.     } 
  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  4.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">  
  5.     <!-- 工廠實例 -- >     
  6.     <bean id="myBeanFactory" class="MyBeanFactory"/>  
  7.     <bean id="factoryInstance" factory-bean="myBeanFactory" factory-method="getInstance"/>  
  8. </beans>  

「使用無參構造函數實例化(默認的)」

  1. public class ConstructorInstance { 
  2.  
  3.     public ConstructorInstance() { 
  4.         System.out.println("ConstructorInstance none args"); 
  5.     } 
  6.  
  1. <?xml version="1.0" encoding="UTF-8"?> 
  2. <beans xmlns="http://www.springframework.org/schema/beans" 
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
  4.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"
  5.     <bean id="constructorInstance" class="spring.service.instance.ConstructorInstance"/> 
  6. </beans> 

「使用有參構造函數實例化」

  1. public class ConstructorInstance { 
  2.  
  3.     private String name
  4.      
  5.     public ConstructorInstance(String name) { 
  6.         System.out.println("ConstructorInstance with args"); 
  7.         this.name = name
  8.     } 
  9.  
  10.     public String getName() { 
  11.         return name
  12.     } 
  13.  
  14.     public void setName(String name) { 
  15.         this.name = name
  16.     } 
  17.  
  1. <?xml version="1.0" encoding="UTF-8"?> 
  2. <beans xmlns="http://www.springframework.org/schema/beans" 
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
  4.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"
  5.         
  6.    <bean id="constructorInstance" class="spring.service.instance.ConstructorInstance"
  7.         <constructor-arg index="0" name="name" value="test constructor with args"/> 
  8.     </bean> 
  9. </beans> 

源碼閱讀

直接來看看doCreateBean方法

具體實現在AbstractAutowireCapableBeanFactory類里面。

我們這里只需關注第一步創建bean實例的流程即可

  1. instanceWrapper = createBeanInstance(beanName, mbd, args); 

上面代碼就是spring 實現bean實例創建的核心代碼。這一步主要根據BeanDefinition里的元數據定義決定使用哪種實例化方法,主要有下面三種:

  • instantiateUsingFactoryMethod 工廠方法實例化的具體實現
  • autowireConstructor 有參構造函數實例化的具體實現
  • instantiateBean 默認實例化具體實現(無參構造函數)

「實例化策略(cglib or 反射)」

❝工廠方法的實例化手段沒有選擇策略直接用了反射實現的,所以這個實例化策略都是對于構造函數實例化而言的❞

下面選一個instantiateBean的實現來介紹


 

 

上面說到的兩構造函數實例化方法不管是哪一種都會選一個實例化策略進行,到底選哪一種策略也是根據BeanDefinition里的定義決定的。

下面這一行代碼就是選擇實例化策略的代碼

  1. beanInstance = getInstantiationStrategy().instantiate(mbd, beanName, parent); 

「選擇使用反射還是cglib」

 

先判斷如果beanDefinition.getMethodOverrides()為空也就是用戶沒有使用replace或者lookup的配置方法,那么直接使用反射的方式,簡單快捷

但是如果使用了這兩個特性,在直接使用反射的方式創建實例就不妥了,因為需要將這兩個配置提供的功能切入進去,所以就必須要使用動態代理的方式將包含兩個特性所對應的邏輯的攔截增強器設置進去,這樣才可以保證在調用方法的時候會被相應的攔截器增強,返回值為包含攔截器的代理實例-----Spring源碼深度解析

  1. <bean id="constructorInstance" class="spring.service.instance.ConstructorInstance" > 
  2.         <lookup-method name="getName" bean="xxx"/> 
  3.         <replaced-method name="getName" replacer="yyy"/> 
  4.     </bean> 

如果使用了lookup或者replaced的配置的話會使用cglib,否則直接使用反射。

  1. public static final String LOOKUP_METHOD_ELEMENT = "lookup-method"
  2.  
  3. public static final String REPLACED_METHOD_ELEMENT = "replaced-method"

覺得不錯,點個贊再走吧,謝謝

參考:

Spring源碼深度解析

Spring In Action

 

https://url.ms/owy8p

本文轉載自微信公眾號「月伴飛魚」,可以通過以下二維碼關注。轉載本文請聯系月伴飛魚公眾號。

 

責任編輯:武曉燕 來源: 月伴飛魚
相關推薦

2012-01-12 14:37:34

jQuery

2011-11-29 12:17:00

2009-02-24 09:43:00

IP電話原理

2016-08-17 23:53:29

網絡爬蟲抓取系統

2021-02-08 21:40:04

SockmapBPF存儲

2011-08-10 19:33:09

Cocoa對象

2019-11-28 10:45:28

ZooKeeper源碼分布式

2016-08-18 00:04:09

網絡爬蟲抓取系統服務器

2010-08-20 13:29:33

OFDM

2013-04-07 14:09:55

Android應用基本

2020-03-21 14:57:14

手機定位智能手機APP

2010-03-17 13:35:02

2011-07-07 14:46:10

Cocoa Xcode

2010-03-18 20:13:03

Java socket

2011-07-07 14:10:21

Cocoa 內省 hash

2009-06-11 09:56:09

MySQL Repli原理

2020-12-29 16:55:44

ZooKeeper運維數據結構

2012-09-28 10:12:55

2010-01-07 09:53:09

Winform多線程編

2024-01-30 09:21:01

deltaFifo機制apiServer
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 精品乱码一区二区三四区视频 | 在线免费中文字幕 | 97久久精品 | 91中文字幕在线 | www.av在线| 久久久久国产精品午夜一区 | 久久网国产| 性天堂网 | 国产精品theporn | 天天夜碰日日摸日日澡 | 欧美激情五月 | 午夜影院官网 | 日韩精品一区二区在线观看 | 欧美日韩高清一区二区三区 | 久久亚洲国产精品日日av夜夜 | 色综合久久天天综合网 | 国产女人与拘做受视频 | 国产免费一级一级 | 日韩在线不卡 | 久久久久久网站 | 中文字幕日韩在线观看 | 91久久综合| 成人一区二 | 97色在线观看免费视频 | 日本免费一区二区三区视频 | 日韩成人在线播放 | 黄色毛片在线看 | 久久久99精品免费观看 | 国产欧美在线观看 | 午夜爱爱毛片xxxx视频免费看 | 欧美一级二级三级 | 天天插天天操 | 成人免费大片黄在线播放 | 欧美成人精品 | 国产精品久久久久久一区二区三区 | av在线免费不卡 | 国产黄色大片在线免费观看 | 日韩欧美在线视频观看 | 国产黑丝av | 国产在线精品一区二区三区 | 日韩精品一区二区三区久久 |