AOP(Aspect Orient Programming)下梢,作為面向?qū)ο缶幊痰囊环N補(bǔ)充,廣泛應(yīng)用于處理一些具有橫切性質(zhì)的系統(tǒng)級服務(wù)塞蹭,如事務(wù)管理孽江、安全檢查、緩存番电、對象池管理等岗屏。AOP 實現(xiàn)的關(guān)鍵就在于 AOP 框架自動創(chuàng)建的 AOP 代理,AOP 代理則可分為靜態(tài)代理和動態(tài)代理兩大類漱办,其中靜態(tài)代理是指使用 AOP 框架提供的命令進(jìn)行編譯这刷,從而在編譯階段就可生成 AOP 代理類,因此也稱為編譯時增強(qiáng)娩井;而動態(tài)代理則在運行時借助于 JDK 動態(tài)代理崭歧、CGLIB 等在內(nèi)存中“臨時”生成 AOP 動態(tài)代理類,因此也被稱為運行時增強(qiáng)撞牢。
先說說AspectJ
在今天之前率碾,我還以為AspectJ 是Spring的一部分,因為我們談到Spring AOP一般都會提到AspectJ屋彪。原來AspectJ是一套獨立的面向切面編程的解決方案所宰。下面我們拋開Spring,單純的看看AspectJ畜挥。
- AspectJ 安裝
AspectJ 下載地址(http://www.eclipse.org/aspectj/downloads.php)仔粥。
下載AspectJ jar包,然后雙擊安裝蟹但。安裝好的目錄結(jié)構(gòu)為:
bin:存放了 aj躯泰、aj5、ajc华糖、ajdoc麦向、ajbrowser 等命令,其中 ajc 命令最常用客叉,它的作用類似于 javac
doc:存放了 AspectJ 的使用說明诵竭、參考手冊话告、API 文檔等文檔
lib:該路徑下的 4 個 JAR 文件是 AspectJ 的核心類庫
- AspectJ HelloWorld 實現(xiàn)
業(yè)務(wù)組件 SayHelloService
package com.ywsc.fenfenzhong.aspectj.learn;
public class SayHelloService {
public void say(){
System.out.print("Hello AspectJ");
}
}
需要來了,在需要在調(diào)用say()方法之后卵慰,需要記錄日志沙郭。那就是通過AspectJ的后置增強(qiáng)吧。
LogAspect 日志記錄組件裳朋,實現(xiàn)對com.ywsc.fenfenzhong.aspectj.learn.SayHelloService 后置增強(qiáng)
package com.ywsc.fenfenzhong.aspectj.learn;
public aspect LogAspect{
pointcut logPointcut():execution(void SayHelloService.say());
after():logPointcut(){
System.out.println("記錄日志 ...");
}
}
- 編譯SayHelloService
執(zhí)行命令 ajc -d . SayHelloService.java LogAspect.java
生成 SayHelloService.class
執(zhí)行命令 java SayHelloService
輸出 Hello AspectJ 記錄日志
ajc.exe 可以理解為 javac.exe 命令病线,都用于編譯 Java 程序,區(qū)別是 ajc.exe 命令可識別 AspectJ 的語法鲤嫡;我們可以將 ajc.exe 當(dāng)成一個增強(qiáng)版的 javac.exe 命令.執(zhí)行
ajc命令后的 SayHelloService.class 文件不是由原來的 SayHelloService.java 文件編譯得到的氧苍,該 SayHelloService.class 里新增了打印日志的內(nèi)容——這表明 AspectJ 在編譯時“自動”編譯得到了一個新類,這個新類增強(qiáng)了原有的 SayHelloService.java 類的功能泛范,因此 AspectJ 通常被稱為編譯時增強(qiáng)的 AOP 框架。
與 AspectJ 相對的還有另外一種 AOP 框架紊撕,它不需要在編譯時對目標(biāo)類進(jìn)行增強(qiáng)罢荡,而是運行時生成目標(biāo)類的代理類,該代理類要么與目標(biāo)類實現(xiàn)相同的接口对扶,要么是目標(biāo)類的子類——總之区赵,代理類的實例可作為目標(biāo)類的實例來使用。一般來說浪南,編譯時增強(qiáng)的 AOP 框架在性能上更有優(yōu)勢——因為運行時動態(tài)增強(qiáng)的 AOP 框架需要每次運行時都進(jìn)行動態(tài)增強(qiáng)笼才。
再談 Spring AOP
Spring AOP也是對目標(biāo)類增強(qiáng),生成代理類络凿。但是與AspectJ的最大區(qū)別在于---Spring AOP的運行時增強(qiáng)骡送,而AspectJ是編譯時增強(qiáng)。
曾經(jīng)以為AspectJ是Spring AOP一部分絮记,是因為Spring AOP使用了AspectJ的Annotation摔踱。使用了Aspect來定義切面,使用Pointcut來定義切入點,使用Advice來定義增強(qiáng)處理怨愤。雖然使用了Aspect的Annotation派敷,但是并沒有使用它的編譯器和織入器。其實現(xiàn)原理是JDK 動態(tài)代理撰洗,在運行時生成代理類篮愉。
為了啟用 Spring 對 @AspectJ 方面配置的支持,并保證 Spring 容器中的目標(biāo) Bean 被一個或多個方面自動增強(qiáng)差导,必須在 Spring 配置文件中添加如下配置
<aop:aspectj-autoproxy/>
當(dāng)啟動了 @AspectJ 支持后试躏,在 Spring 容器中配置一個帶 @Aspect 注釋的 Bean,Spring 將會自動識別該 Bean设褐,并將該 Bean 作為方面 Bean 處理冗酿。方面Bean與普通 Bean 沒有任何區(qū)別埠对,一樣使用 <bean.../> 元素進(jìn)行配置,一樣支持使用依賴注入來配置屬性值裁替。
使用Spring AOP的改寫 Hello World的例子项玛。
業(yè)務(wù)組件 SayHelloService
package com.ywsc.fenfenzhong.aspectj.learn;
import org.springframework.stereotype.Component;
@Component
public class SayHelloService {
public void say(){
System.out.print("Hello AspectJ");
}
}
做后置增強(qiáng)的日志處理。
package com.ywsc.fenfenzhong.aspectj.learn;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
@Aspect
@Component
public class LogAspect {
@After("execution(* com.ywsc.fenfenzhong.aspectj.learn.SayHelloService.*(..))")
public void log(){
System.out.println("記錄日志 ...");
}
}
package com.ywsc.fenfenzhong.mongodb;
import com.ywsc.fenfenzhong.aspectj.learn.SayHelloService;
public class TestCase {
public static void main(String[] args) {
SayHelloService sayHelloService = ApplicationUtil.getContext().getBean(SayHelloService.class);
sayHelloService.say();
}
}
輸出結(jié)果
Hello AspectJ
記錄日志...```
####Hello World后的總結(jié)
AOP 代理 = 原來的業(yè)務(wù)類+增強(qiáng)處理弱判。
這個生成AOP 代理由 Spring 的 IoC 容器負(fù)責(zé)生成襟沮。也由 IoC 容器負(fù)責(zé)管理。因此昌腰,AOP 代理可以直接使用容器中的其他 Bean 實例作為目標(biāo)开伏,這種關(guān)系可由 IoC 容器的依賴注入提供≡馍蹋回顧Hello World的例子固灵,其中程序員參與的只有 3 個部分:
. 定義普通業(yè)務(wù)組件。
. 定義切入點劫流,一個切入點可能橫切多個業(yè)務(wù)組件巫玻。
. 定義增強(qiáng)處理,增強(qiáng)處理就是在 AOP 框架為普通業(yè)務(wù)組件織入的處理動作祠汇。
###最后說說CGLIB
CGLIB(Code Generation Library)它是一個代碼生成類庫仍秤。它可以在運行時候動態(tài)是生成某個類的子類。代理模式為要訪問的目標(biāo)對象提供了一種途徑可很,當(dāng)訪問對象時诗力,它引入了一個間接的層。JDK自從1.3版本開始我抠,就引入了動態(tài)代理苇本,并且經(jīng)常被用來動態(tài)地創(chuàng)建代理。JDK的動態(tài)代理用起來非常簡單菜拓,唯一限制便是使用動態(tài)代理的對象必須實現(xiàn)一個或多個接口圈澈。而CGLIB缺不必有此限制。要想Spring AOP 通過CGLIB生成代理尘惧,只需要在Spring 的配置文件引入
```xml
<aop:aspectj-autoproxy proxy-target-class="true"/>
CGLIB包的底層是通過使用一個小而快的字節(jié)碼處理框架ASM(Java字節(jié)碼操控框架)康栈,來轉(zhuǎn)換字節(jié)碼并生成新的類。由于沒有了解過class 文件和字節(jié)碼喷橙,因而也就寫不下去了啥么。
也許學(xué)習(xí)下來最大的收獲便是 弄清楚了 AspectJ 和 Spring AOP 在實現(xiàn)上幾乎無關(guān)。