<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    莊周夢蝶

    生活、程序、未來
       :: 首頁 ::  ::  :: 聚合  :: 管理

    Spring源碼分析:實現AOP(轉載)

    Posted on 2007-04-24 09:37 dennis 閱讀(7540) 評論(0)  編輯  收藏 所屬分類: java 、源碼解讀
        這兩天一直在讀spring1.2的AOP實現源碼,AOP實現原理說起來很簡單,對于實現業務接口的對象使用java代理機制實現,而對于一般的類使用cglib庫實現,但spring的實現還是比較復雜的,不過抓住了本質去看代碼就容易多了。發現一篇04年寫的《spring源碼分析:實現AOP》,倒是不用自己再寫了,04年的時候已經有很多人研讀過spring的源碼,而那時的我還在學校,對java半懂不懂的狀態,就算到現在也不敢說真的懂了,繼續學習、努力。文章如下:

       
    我的問題
           為了完成公司應用開發平臺的設計,這幾天一直在研究Spring的擴展機制。Spring的核心無疑是BeanFactory, ApplicationContext和AOP。在“Spring AOP編程”教程的例子中,是由ProxyFactoryBean來實現的。問題來了,普通的bean和FactoryBean的配置完全是一樣的。那 么,BeanFactory是如何區分普通的Bean和用作Proxy的FactoryBean的?ProxyFactoryBean又是怎樣實現AOP 功能的?(本文還很不完善,我會繼續修改。)
     
    FactoryBean的職責
            FactoryBean在Spring中被當成一種特殊的bean,通過實現FactoryBean接口進行擴展。FactoryBean的職責是:
            l.封裝了創建對象或查找對象的邏輯。
           2.提供了一個中間層,用于支持AOP。

           我們來看一個LocalStatelessSessionProxyFactoryBean的例子。首先,定義Stateless EJB的代理,id為ejbServiceProxy:
           <bean id="ejbServiceProxy" class="LocalStatelessSessionProxyFactoryBean">
               <property name="jndiName">     
                  <value>myEjb</value>
               </property>
               <property name="businessInterface">
                  <value>com.mycompany.MyBusinessInterface</value>
               </property>
          </bean>
     
          然后,再將這個業務邏輯服務對象注入客戶程序:
         <bean id="myAction" class = "samples.myAction">
             <property name="myService">
                 <ref bean="ejbServiceProxy"/>
             </property>
         </bean>

         這樣,客戶程序并不知道myService的實現細節,Spring使用FactoryBean完成了兩者之間的解耦。
     
    準備代碼分析環境
         1. 安裝Eclipse和Spring IDE。
         2. 下載Spring framework源代碼,并導入Eclipse。
         3. 在類路徑創建log4j.properties配置文件,設置如下:
                 log4j.rootLogger=DEBUG, stdout
                 log4j.appender.stdout=org.apache.log4j.ConsoleAppender
                 log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
                 log4j.appender.stdout.layout.ConversionPattern=%d{SSS} %p %c{2} - %m%n
         4. 編寫TestCase,跟蹤Console窗口的debug信息。

    FactoryBean源代碼分析
           如果bean實現了FactoryBean接口,BeanFactory將把它作為一個bean工廠,而不是直接作為普通的bean。正常情況下, BeanFactory的getBean("bean")返回FactoryBean生產的bean實例,如果要返回FactoryBean本身的實例, 使用getBean("&bean")的調用方式。
           在分析ProxyFactoryBean之前,我們先分析BeanFactory,它是Spring Framework的基礎。我們看看它是如何分別處理普通的Bean和FactoryBean的。
     
      BeanFactory分析
     
         BeanFactory類圖
     
           如以上的類圖所示,XmlBeanFactory繼承了AbstactBeanFactory抽象類。AbstactBeanFactory類中使用了 Template Method設計模式,其中的模板方法為getBeanDefinition()和createBean()兩個抽象方法。其中 AbstractAutowireCapableBeanFactory類實現了getBeanDefinition()方法, DefaultAutowireCapableBeanFactory類實現了getBeanDefinition()方法。當調用getBean()方 法時,AbstractBeanFactory類定義的邏輯分別調用了這兩個模板方法。

         BeanFactory類的調用順序
           我們暫時不使用ApplicationContext,以簡化分析過程。我在這里使用了“Spring AOP編程”的例子,請參照該教程閱讀。首先,編寫測試用例,代碼如下:

                public class AopTest extends TestCase {
                      XmlBeanFactory factory = null;
                      protected void setUp() throws Exception {
                           super.setUp();
                           InputStream is = new FileInputStream("testaop.xml");
                           factory = new XmlBeanFactory(is);
                      }
                      public void testGetBean() {
                           Bean bean = (Bean)factory.getBean("bean");
                           assertNotNull(bean);
                           bean.theMethod();
                      }
                }
     
           1. 首先,XmlBeanFactory使用XmlBeanDefinitionReader讀入testaop.xml配置文件,后者用 XmlBeanDefinitionParser和DefaultXmlBeanDefinitionParser進行分析,從中得到 BeanDefinition的信息,并保存在XmlBeanDefinitionReader的BeanDefinitionRegistry變量里。
           2. 客戶程序調用getBean方法時,AbstractBeanFactory首先使用transFormedBeanName方法分析傳入的Bean名稱,判斷客戶程序需要FactoryBean本身,還是它所創建的Bean對象。
           3. 接下來,如果bean被定義為singleton模式,AbstractBeanFactory調用createBean方法根據 BeanDefinition信息實例化bean類,然后將該bean實例傳給getObjectForSharedInstance方法并返回 getObjectForSharedInstance的返回對象。GetObjectForSharedInstance方法摘要如類圖所示,首先判斷 bean是否繼承了FactoryBean。如果是,返回FactoryBean的getObject方法(下節我將詳細分析使用 ProxyFactoryBean如何實現AOP);如果不是,返回bean對象。
           4. 如果bean被定義為prototype模式,每次客戶程序請求都會生成新的bean實例,因此,createBean方法直接實例化bean對象并返回。
     
      ProxyFactoryBean如何實現AOP

         ProxyFactoryBean類圖
           FactoryBean接口如下圖所示,共有三個方法getObject,getObjectType,和isSingleton。ProxyFactoryBean實現了FactoryBean接口,它的相關類圖如下:
     

     
         實現AOP的過程
           如上圖所示,ProxyFactoryBean類繼承了AdvisedSupport類,后者繼承了ProxyConfig類并定義了操作advisor 和interceptor的接口,以支持AOP。當BeanFactory實例化ProxyFactoryBean時,根據配置文件的定義將關于 advice,pointcut,advisor,所代理的接口和接口實現類的所有信息傳給ProxyFactoryBean。
           當客戶程序調用BeanFactory的getBean方法時,ProxyFactory使用JdkDynamicAopProxy實例化 BeanImpl類,并用JdkDynamicAopProxy的invoke方法執行advice。至于執行advice的時機,由 ProxyFactoryBean調用RegexpMethodPointcutAdvisor進行判斷。


    主站蜘蛛池模板: 久久久WWW免费人成精品| 国产特级淫片免费看| 91成年人免费视频| 日韩一区二区三区免费体验| 亚洲午夜精品第一区二区8050| 久久久无码精品亚洲日韩蜜桃 | 国产美女无遮挡免费网站| 亚洲天堂中文字幕在线| 亚洲综合在线成人一区| 国产99久久久久久免费看| 国产在线jyzzjyzz免费麻豆| 日韩精品视频免费在线观看| 亚洲av综合avav中文| 国产亚洲精品美女久久久久久下载| 免费人成在线观看网站| 精品久久久久久久免费加勒比| 亚洲精品国产品国语在线| 亚洲av无码专区亚洲av不卡| 99re在线视频免费观看| 日韩亚洲变态另类中文| 免费人成网站永久| 国产a不卡片精品免费观看| 国产91在线|亚洲| 中文字幕在线观看免费视频| 国产亚洲精品国看不卡| 美女18一级毛片免费看| 最近最新中文字幕完整版免费高清| 久久亚洲精品成人综合| 国产一精品一av一免费爽爽 | 成年女人喷潮毛片免费播放| 亚洲午夜成激人情在线影院| 91精品免费久久久久久久久| 激情内射亚洲一区二区三区| 久草免费福利视频| 亚洲VA中文字幕无码一二三区 | 男人的天堂av亚洲一区2区| 日本不卡免费新一二三区| 国产成人精品日本亚洲语音| 免费大黄网站在线观看| 四虎影视久久久免费| 亚洲av中文无码乱人伦在线播放|