理解Spring AOP的基本概念和接口

前言

這篇文章主要將一下關於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能夠看做兩步:
  1. 根據配置文件或者註解生成代理對象
  2. 執行代理對象,代理對象中包括加強方法和被代理對象
下面簡單瞭解下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實現
相關文章
相關標籤/搜索