學習AOP之深入一點Spring Aop

来源:http://www.cnblogs.com/5207/archive/2016/11/11/6055152.html
-Advertisement-
Play Games

上一篇 "《學習AOP之認識一下SpringAOP》" 中大體的瞭解了代理、動態代理及SpringAop的知識。因為寫的篇幅長了點所以還是再寫一篇吧。接下來開始深入一點Spring aop的一些實現機制。 上篇中最後有那段代碼使用了一個ProxyFactory類來完成代理的工作,從而實現了Aop的A ...


上一篇《學習AOP之認識一下SpringAOP》中大體的瞭解了代理、動態代理及SpringAop的知識。因為寫的篇幅長了點所以還是再寫一篇吧。接下來開始深入一點Spring aop的一些實現機制。

上篇中最後有那段代碼使用了一個ProxyFactory類來完成代理的工作,從而實現了Aop的Around Advice,代碼如下:

package aop.demo;

import org.springframework.aop.framework.ProxyFactory;

public class ClientCode {

    public static void main(String[] args) {
        ProxyFactory proxyFactory = new ProxyFactory();     // 創建代理工廠
        proxyFactory.setTarget(new SayImpl());         // 射入目標類對象
        proxyFactory.addAdvice(new SayImplAroundAdvice());
        ISay say = (ISay) proxyFactory.getProxy();
        say.say();
    }

}

那麼接下來就聊聊ProxyFactory吧,看看它都幹了些啥。

1、ProxyFactory的奧秘

繼續看上面的代碼只用了5行,這裡面意思也非常明確,只有在第4行的時候有一個getProxy的方法並轉換為ISay介面。看來代理對象的來源可以從它入手了。

public Object getProxy() {
    return createAopProxy().getProxy();
}

只不過代碼只有一行,調用的是一個createAopProxy()的方法返回了AopProxy類型的對象,再通過AopProxy的getProxy來獲得了代理對象。

那麼只好再看一下createAopProxy()是啥樣子咯:

protected final synchronized AopProxy createAopProxy() {
    if (!this.active) {
        activate();
    }
    return getAopProxyFactory().createAopProxy(this);
}

這個方法在org.springframework.aop.framework.ProxyCreatorSupport這個類裡面,ProxyFactory是繼承了它的。這個類字面意思就是一個代理創建的支持類。

但是看了createAopProxy方法後又鬱悶了,還有一個getAopProxyFactory(),真是一層套一層啊。當然這裡還是需要從類的層次結構來看會清楚一些,只是我主要是看它是怎麼生成代理對象的,設計上的事情回頭再看。

//這個方法訪問了一個內部成員
public AopProxyFactory getAopProxyFactory() {
    return this.aopProxyFactory;
}

//再看aopProxyFactory其實是在構造函數里創建的
public ProxyCreatorSupport() {
    this.aopProxyFactory = new DefaultAopProxyFactory();
}

這裡看到了DefaultAopProxyFactory這個工廠類,好了,也就是它才是創建代理的真正人物。那麼這裡接著createAopProxy直接看代碼:

public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
    if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) {
        Class targetClass = config.getTargetClass();
        if (targetClass == null) {
            throw new AopConfigException("TargetSource cannot determine target class: " +
                    "Either an interface or a target is required for proxy creation.");
        }
        if (targetClass.isInterface()) {
            return new JdkDynamicAopProxy(config);
        }
        return CglibProxyFactory.createCglibProxy(config);
    }
    else {
        return new JdkDynamicAopProxy(config);
    }
}

從這裡可以看到有兩種AopProxy的代理:Cglib和Jdk。它們倆的區別:

  • Cglib創建代理慢但執行快而且可以代理類
  • Jdk創建代理快但執行慢,只可以代理介面

順著ClientCode這個代碼肯定是用JdkDynmaicAopProxy,最終proxyFactory.getProxy()調用的是JdkDynmaicAopProxy的實例。那好就看一下JdkDynmaicAopProxy中getProxy都做了啥吧:

public Object getProxy() {
    return getProxy(ClassUtils.getDefaultClassLoader());
}

public Object getProxy(ClassLoader classLoader) {
    if (logger.isDebugEnabled()) {
        logger.debug("Creating JDK dynamic proxy: target source is " + this.advised.getTargetSource());
    }
    Class[] proxiedInterfaces = AopProxyUtils.completeProxiedInterfaces(this.advised);
    findDefinedEqualsAndHashCodeMethods(proxiedInterfaces);
    return Proxy.newProxyInstance(classLoader, proxiedInterfaces, this);
}

好了,這裡看到了熟悉的代碼,即通過Proxy.newProxyInstance生成代理對象交給調用者。Spring通過抽象工廠模式設計了兩種代理方法。

2、再看看ProxyFactroyBean

但是在xml配置的時候用的並不是ProxyFactory,而是ProxyFactroyBean。有點奇怪,為什麼會有兩個類呢?先來看看ProxyFactoryBean:

public class ProxyFactoryBean extends ProxyCreatorSupport
        implements FactoryBean<Object>, BeanClassLoaderAware, BeanFactoryAware {

哦喲,原來這家伙繼承了FactoryBean,好了,原來它借且FactoryBean提供了一個中間層。因為Spring如果發現Ioc創建的對象帶有FactoryBean時會調用FactoryBean的getObject方法來獲得對象。有了這個它在Ioc容器里獲得的便是getObject返回的代理對象,而不是返回ProxyFactoryBean本身,這樣才能註入嘛。所以ProxyFactoryBean主要是使用在Ioc容器里的。

具體getObject里實現的原理和ProxyFactory類似,主要還是和ProxyCreatorSupport有關,ProxyCreatorSupport封裝了這部分邏輯,所以可以復用。

3、進入切麵的小世界

寫了這麼多發現我還是停留在“代理”的層面,但是AOP難道僅僅止於此嗎?當然不是,比如ISay介面增加一個noaop方法,這個方法我就不希望被代理,那怎麼做呢?

先調整一下例子代碼,增加noaop方法。

public interface ISay {
    void say();
    void noaop();
}

public class SayImpl implements ISay{

    public void say() {
        System.out.print("我是5207.");
    }

    public void noaop() {
        System.out.println("別aop我");
    }

}

好了,然後增加一個切麵,讓這個切麵去做分辨,以xml配置為例,下麵對spring.xml做一下修改:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       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">

    <!-- 聲明被代理的目標對象 -->
    <bean id="sayImpl" class="aop.demo.SayImpl"></bean>
    <!-- 聲明用於增強的攔截器對象 -->
    <bean id="sayImplAroundAdvice" class="aop.demo.SayImplAroundAdvice"></bean>

    <!-- 配置一個切麵 -->
    <bean id="sayAdvisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
        <property name="advice" ref="sayImplAroundAdvice"/>            <!-- 增強 -->
        <property name="pattern" value="aop.demo.SayImpl.s.*"/> <!-- 切點(正則表達式) -->
    </bean>   
    
    <!-- 聲明代理對象 -->
    <bean id="sayProxy" class="org.springframework.aop.framework.ProxyFactoryBean">
        <property name="interfaces" value="aop.demo.ISay"/>            <!-- 這個就是被代理的介面 -->
        <property name="target" ref="sayImpl"/>                        <!-- 這個就是被代理的對象 -->
        <property name="interceptorNames" value="sayAdvisor"/>         <!-- 這個就是代理的增強器 --> 
    </bean>
</beans>

上面xml中新增了一個切麵sayAdvisor,它的作用是以正則表達式的規則來選擇是否aop。比如本例子的意思是只代理SyaImpl的s開頭的方法。那麼noaop方法應該是不會被代理啦。

client代碼也修改一下:

public class Client {

    @SuppressWarnings("resource")
    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("aop/demo/spring.xml");
        ISay say = (ISay)context.getBean("sayProxy");
        say.say();
        say.noaop();//增加noaop的調用,看看會不會被代理
    }

}

執行的結果如下:

大家好:我是5207.希望大家多多點贊.
別aop我

這就發現advisor已經有效果啦。

4、自動完成對切麵的代理

之前的各種代碼都帶有一個問題,就是client最終調用的時候都是獲得的代理對象,如下麵的代碼:

ISay say = (ISay)context.getBean("sayProxy");

那在做Aop增強的時候改老的代碼,這樣就失敗了Aop的意義了。所以沒有辦法可以自動就完成這個操作,只要配置好就可以透明的完成這個代理過程呢?

spring提供了自動代理的實現,對spring.xml做一下調整:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       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">

    <!-- 聲明被代理的目標對象 -->
    <bean id="sayImpl" class="aop.demo.SayImpl"></bean>
    <!-- 聲明用於增強的攔截器對象 -->
    <bean id="sayImplAroundAdvice" class="aop.demo.SayImplAroundAdvice"></bean>

    <!-- 配置一個切麵 -->
    <bean id="sayAdvisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor">
        <property name="advice" ref="sayImplAroundAdvice"/>            <!-- 增強 -->
        <property name="pattern" value="aop.demo.SayImpl.s.*"/> <!-- 切點(正則表達式) -->
    </bean>   
   
    <bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator">
        <property name="optimize" value="true"/>
    </bean>
</beans>

在此增加一個DefaultAdvisorAutoProxyCreator,原先的代理就不需要啦。然後再看一下客戶端調用直接改成調用SayImpl,看看能不能實現代理:

package aop.demo;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class Client {

    @SuppressWarnings("resource")
    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("aop/demo/spring.xml");
        ISay say = (ISay)context.getBean("sayImpl");
        say.say();
        say.noaop();
    }
}

輸出:

大家好:我是5207.希望大家多多點贊.
別aop我

效果達成,只不過這裡的關鍵是DefaultAdvisorAutoProxyCreater是怎麼做的呢?看了看代碼發現其主要是藉助Ioc容器在初始化對象時完成的代理的自動生成的。在BeanPostProccer的postProcessAfterInitialization過程中完成了對代理的生成。具體的原理可以參考引用中的文章,太累了不寫了。

參考及引用
死磕Spring AOP系列2:剖析Bean處理器之BeanNameAutoProxyCreator

註:此文章為原創,歡迎轉載,請在文章頁面明顯位置給出此文鏈接!
若您覺得這篇文章還不錯請點擊下右下角的推薦,非常感謝!
http://www.cnblogs.com/5207


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 很多新手一聽到介面就蒙逼,不知道介面是什麼!其實介面就是RPC,通過遠程訪問別的程式提供的方法,然後獲得該方法執行的介面,而不需要在本地執行該方法。就是本地方法調用的升級版而已,我明天會上一篇如何通過socket實現rpc,以及服務的註冊和動態上下線。這裡先上一篇RPC的實現者一webservice ...
  • CListCtrl CListCtrl類封裝“列表視圖控制項”功能,顯示每個包含圖標(列表視圖中)和標簽的收集。除圖標和標簽外,每一項還能有顯示在圖標和標簽的右邊的列中的信息。視圖 列表視圖控制項可用四種不同方式顯示其內容,稱為“視圖”。 擴展風格 除了標準列表風格,類CListCtrl支持一系列提供豐 ...
  • 通過Xdebug進行遠程調試時,PHPSTORM等IDE會監聽Xdebug設置的調試ip和埠(php.ini中配置) 下麵這張截圖是phpstorm開啟listening for php debug connections 後: 併在當前調試的URL後面加上了XDEBUG_SESSION_STAR ...
  • 看《Java併發編程實戰》遇到如下問題 代碼: public void assertSanity(); flags: ACC_PUBLIC Code: stack=3, locals=1, args_size=1 0: aload_0 1: getfield #2 // Field n:I 4: a ...
  • 面向對象最具特色的“繼承”環節,其中諸多知識點將一一提到。 單個繼承,但是不失多重繼承的靈活性,以“介面”替代之,降低多個類之間關係複雜度。 ...
  • webService 可以將應用程式轉換成網路應用程式。是簡單的可共同操作的消息收發框架。 基本的webService平臺是 XML 和 HTTP。 HTTP 是最常用的互聯網協議; XML 是 webService 的基礎,因為可以用於不同平臺和編程語言之間。 webService平臺的元素: S ...
  • LPTSTR、LPCSTR、LPSTR、LPCTSTR、LPWSTR、LPCWSTR: 具體查看:http://blog.csdn.net/yibo_ge/article/details/51058917> http://www.cppblog.com/gezidan/archive/2011/08 ...
  • Qt的網路操作類是非同步(非阻塞的),但有時想做一些阻塞的事情就不方便了,可用如下幾行代碼輕鬆實現: 當然如上方式不支持重定向(301等),因為暫時用不上,如果要支持,還要在return前判斷並迴圈或遞歸。 另外如果出現error,現在的方式會把伺服器返回的錯誤信息直接返回,後面再更新一版,支持判斷錯 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...