前言
這篇文章主要將一下關於Spring中AOP的基本概念和相關的接口,也算是對我本身以前學習Spring的回顧與總結。
AOP是什麼
AOP全稱是Aspect Oriented Programming,面向切面編程,是面向對象編程(OOP:Object Oriented Programming)的補充和完善。通常在系統中,OOP利用繼承、封裝、多態來模擬處理業務模塊和邏輯;AOP則是封裝非業務但又被各業務模塊頻繁調用的功能,如日誌、權限認證、事物。利用AOP可有效減小系統中的重複代碼、下降模塊間的耦合、利於維護。java
AOP 相關術語
在咱們開始使用 AOP 工做以前,讓咱們熟悉一下 AOP 概念和術語。這些術語並不特定於 Spring,而是與 AOP 有關的。

1)鏈接點(Joinpoint)
程序執行的某個特定位置:如類開始初始化前、類初始化後、類某個方法調用前、調用
後、方法拋出異常後。一個類或一段程序代碼擁有一些具備邊界性質的特定點,這些點中的
特定點就稱爲「鏈接點」。Spring 僅支持方法的鏈接點,即僅能在
方法調用前、
方法調用
後、
方法拋出異常時以及
方法調用先後這些程序執行點織入加強。
2)切點(Pointcut)
每一個程序類都擁有多個鏈接點,如一個擁有兩個方法的類,這兩個方法都是鏈接點,即
鏈接點是程序類中客觀存在的事物。AOP 經過「切點」定位特定的鏈接點。
3)通知或加強(Advice)
加強是織入到目標類鏈接點上的一段程序代碼,在Spring 中,加強除用於描述一段程
序代碼外,還擁有另外一個和鏈接點相關的信息,這即是執行點的方位。結合執行點方位信息
和切點信息,咱們就能夠找到特定的鏈接點。
Spring 中定義瞭如下幾種通知類型:編程
- 前置通知(Before advice)- 在目標方便調用前執行通知
- 後置通知(After advice)- 在目標方法完成後執行通知
- 返回通知(After returning advice)- 在目標方法執行成功後,調用通知
- 異常通知(After throwing advice)- 在目標方法拋出異常後,執行通知
- 環繞通知(Around advice)- 在目標方法調用先後都可執行自定義邏輯
4)目標對象(Target)
加強邏輯的織入目標類。若是沒有AOP,目標業務類須要本身實現全部邏輯,而在AOP
的幫助下,目標業務類只實現那些非橫切邏輯的程序邏輯,而性能監視和事務管理等這些橫
切邏輯則可使用AOP 動態織入到特定的鏈接點上。
5)引介(Introduction)
引介是一種特殊的加強,它爲類添加一些屬性和方法。這樣,即便一個業務類本來沒有
實現某個接口,經過AOP 的引介功能,咱們能夠動態地爲該業務類添加接口的實現邏輯,
讓業務類成爲這個接口的實現類。
6)織入(Weaving)
織入是將加強添加對目標類具體鏈接點上的過程。AOP 像一臺織布機,將目標類、增
強或引介經過AOP 這臺織布機完美無缺地編織到一塊兒。根據不一樣的實現技術,AOP 有三種
織入的方式:
a、編譯期織入,這要求使用特殊的Java 編譯器。
b、類裝載期織入,這要求使用特殊的類裝載器。
c、動態代理織入,在運行期爲目標類添加加強生成子類的方式。
Spring 採用動態代理織入,而AspectJ 採用編譯期織入和類裝載期織入。
7)代理(Proxy)
一個類被AOP 織入加強後,就產出了一個結果類,它是融合了原類和加強邏輯的代理
類。根據不一樣的代理方式,代理類既多是和原類具備相同接口的類,也可能就是原類的子
類,因此咱們能夠採用調用原類相同的方式調用代理類。
8)切面(Aspect)
切面由切點和加強(引介)組成,它既包括了橫切邏輯的定義,也包括了鏈接點的定義,
Spring AOP 就是負責實施切面的框架,它將切面所定義的橫切邏輯織入到切面所指定的
鏈接點中。
核心接口類
整個Spring AOP能夠看做兩步:
- 根據配置文件或者註解生成代理對象
- 執行代理對象,代理對象中包括加強方法和被代理對象
下面簡單瞭解下Spring AOP中核心的接口類
InvocationHandler
接口,從字面不難理解是調用處理器的意思,該接口是一個祖宗級接口,大多動態代理接口都會實現它,如JdkDynamicAopProxy。其中只定義的了一個方法
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable;
複製代碼
AopProxy
接口,也是一個祖宗級接口,定義代理。只有三個實現類CglibAopProxy、JdkDynamicAopProxy、ObjenesisCglibAopProxy,接口內部只定義了兩個方法
Object getProxy();
Object getProxy(ClassLoader classLoader);複製代碼
AopProxyFactory
接口,生成代理的工廠。接口只有一個方法,定義了建立代理的方法,實現以下:
public interface AopProxyFactory {
AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException;
}複製代碼
在Spring AOP中,
該接口只有一個實現類DefaultAopProxyFactory。DefaultAopProxyFactory實現了接口中的createAopProxy方法,定義了什麼時候使用JdkDynamicAopProxy、什麼時候使用ObjenesisCglibAopProxy代理。源碼以下:
public class DefaultAopProxyFactory implements AopProxyFactory, Serializable {
@Override
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() || Proxy.isProxyClass(targetClass)) {
return new JdkDynamicAopProxy(config);
}
return new ObjenesisCglibAopProxy(config);
}
else {
return new JdkDynamicAopProxy(config);
}
}
private boolean hasNoUserSuppliedProxyInterfaces(AdvisedSupport config) {
Class<?>[] ifcs = config.getProxiedInterfaces();
return (ifcs.length == 0 || (ifcs.length == 1 && SpringProxy.class.isAssignableFrom(ifcs[0])));
}
}複製代碼
從實現中能夠看出Spring AOP中默認是應用JDK動態代理實現AOP功能,
當代理對象爲接口時使用JDK動態代理,
不然使用ObjenesisCglibAopProxy實現。