Spring AOP底層機制之代理模式

代理模式

為什麼要學習代理模式,因為AOP的底層機制就是動態代理!

代理模式:

  • 靜態代理
  • 動態代理

學習aop之前 , 我們要先瞭解一下代理模式!

靜態代理

靜態代理角色分析

  • 抽象角色 : 一般使用接口或者抽象類來實現
  • 真實角色 : 被代理的角色
  • 代理角色 : 代理真實角色 ; 代理真實角色後 , 一般會做一些附屬的操作 .
  • 客戶 : 使用代理角色來進行一些操作 .

代碼實現

Rent . java 即抽象角色

public interface Rent {
    public void rent();
}

Host . java 即真實角色

//真實角色: 房東,房東要出租房子
public class Host implements Rent{
   public void rent() {
       System.out.println("房屋出租");
  }
}

Proxy . java 即代理角色

//代理角色:中介
public class Proxy implements Rent {
   private Host host;
   public Proxy() { }
   public Proxy(Host host) {
       this.host = host;
  }
   //租房
   public void rent(){
       seeHouse();
       host.rent();
       fare();
  }
   //看房
   public void seeHouse(){
       System.out.println("帶房客看房");
  }
   //收中介費
   public void fare(){
       System.out.println("收中介費");
  }
}

Client . java 即客戶

//客戶類,一般客戶都會去找代理!
public class Client {
   public static void main(String[] args) {
       //房東要租房
       Host host = new Host();
       //中介幫助房東
       Proxy proxy = new Proxy(host);
       //你去找中介!
       proxy.rent();
  }
}

分析:在這個過程中,你直接接觸的就是中介,就如同現實生活中的樣子,你看不到房東,但是你依舊租到瞭房東的房子通過代理,這就是所謂的代理模式,程序源自於生活,所以學編程的人,一般能夠更加抽象的看待生活中發生的事情。

靜態代理的好處:

  • 可以使得我們的真實角色更加純粹 . 不再去關註一些公共的事情 .
  • 公共的業務由代理來完成 . 實現瞭業務的分工 ,
  • 公共業務發生擴展時變得更加集中和方便 .

缺點 :

類多瞭 , 多瞭代理類 , 工作量變大瞭 . 開發效率降低 .

我們想要靜態代理的好處,又不想要靜態代理的缺點,所以 , 就有瞭動態代理 !

靜態代理再理解

同學們練習完畢後,我們再來舉一個例子,鞏固大傢的學習!

練習步驟:

1、創建一個抽象角色,比如咋們平時做的用戶業務,抽象起來就是增刪改查!

//抽象角色:增刪改查業務
public interface UserService {
   void add();
   void delete();
   void update();
   void query();
}

2、我們需要一個真實對象來完成這些增刪改查操作

//真實對象,完成增刪改查操作的人
public class UserServiceImpl implements UserService {
   public void add() {
       System.out.println("增加瞭一個用戶");
  }
   public void delete() {
       System.out.println("刪除瞭一個用戶");
  }
   public void update() {
       System.out.println("更新瞭一個用戶");
  }
   public void query() {
       System.out.println("查詢瞭一個用戶");
  }
}

3、需求來瞭,現在我們需要增加一個日志功能,怎麼實現!

  • 思路1 :在實現類上增加代碼 【麻煩!】
  • 思路2:使用代理來做,能夠不改變原來的業務情況下,實現此功能就是最好的瞭!

4、設置一個代理類來處理日志!代理角色

//代理角色,在這裡面增加日志的實現
public class UserServiceProxy implements UserService {
   private UserServiceImpl userService;
   public void setUserService(UserServiceImpl userService) {
       this.userService = userService;
  }
   public void add() {
       log("add");
       userService.add();
  }
   public void delete() {
       log("delete");
       userService.delete();
  }
   public void update() {
       log("update");
       userService.update();
  }
   public void query() {
       log("query");
       userService.query();
  }
   public void log(String msg){
       System.out.println("執行瞭"+msg+"方法");
  }
}

5、測試訪問類:

public class Client {
   public static void main(String[] args) {
       //真實業務
       UserServiceImpl userService = new UserServiceImpl();
       //代理類
       UserServiceProxy proxy = new UserServiceProxy();
       //使用代理類實現日志功能!
       proxy.setUserService(userService);
       proxy.add();
  }
}

OK,到瞭現在代理模式大傢應該都沒有什麼問題瞭,重點大傢需要理解其中的思想;

我們在不改變原來的代碼的情況下,實現瞭對原有功能的增強,這是AOP中最核心的思想

聊聊AOP:縱向開發,橫向開發

動態代理

動態代理的角色和靜態代理的一樣 .

動態代理的代理類是動態生成的 . 靜態代理的代理類是我們提前寫好的

動態代理分為兩類 : 一類是基於接口動態代理 , 一類是基於類的動態代理

  • 基於接口的動態代理—-JDK動態代理
  • 基於類的動態代理–cglib
  • 現在用的比較多的是 javasist 來生成動態代理 . 百度一下javasist
  • 我們這裡使用JDK的原生代碼來實現,其餘的道理都是一樣的!

JDK的動態代理需要瞭解兩個類

核心 : InvocationHandler 和 Proxy , 打開JDK幫助文檔看看

【InvocationHandler:調用處理程序】

Object invoke(Object proxy, 方法 method, Object[] args);
//參數
//proxy - 調用該方法的代理實例
//method -所述方法對應於調用代理實例上的接口方法的實例。方法對象的聲明類將是該方法聲明的接口,它可以是代理類繼承該方法的代理接口的超級接口。
//args -包含的方法調用傳遞代理實例的參數值的對象的陣列,或null如果接口方法沒有參數。原始類型的參數包含在適當的原始包裝器類的實例中,例如java.lang.Integer或java.lang.Boolean 。

【Proxy : 代理】

//生成代理類
public Object getProxy(){
   return Proxy.newProxyInstance(this.getClass().getClassLoader(),
                                 rent.getClass().getInterfaces(),this);
}

代碼實現

抽象角色和真實角色和之前的一樣!

Rent . java 即抽象角色

//抽象角色:租房
public interface Rent {
   public void rent();
}

Host . java 即真實角色

//真實角色: 房東,房東要出租房子
public class Host implements Rent{
   public void rent() {
       System.out.println("房屋出租");
  }
}

ProxyInvocationHandler. java 即代理角色

public class ProxyInvocationHandler implements InvocationHandler {
   private Rent rent;
   public void setRent(Rent rent) {
       this.rent = rent;
  }
   //生成代理類,重點是第二個參數,獲取要代理的抽象角色!之前都是一個角色,現在可以代理一類角色
   public Object getProxy(){
       return Proxy.newProxyInstance(this.getClass().getClassLoader(),
               rent.getClass().getInterfaces(),this);
  }
   // proxy : 代理類 method : 代理類的調用處理程序的方法對象.
   // 處理代理實例上的方法調用並返回結果
   @Override
   public Object invoke(Object proxy, Method method, Object[] args) throwsThrowable {
       seeHouse();
       //核心:本質利用反射實現!
       Object result = method.invoke(rent, args);
       fare();
       return result;
  }
   //看房
   public void seeHouse(){
       System.out.println("帶房客看房");
  }
   //收中介費
   public void fare(){
       System.out.println("收中介費");
  }
}

Client . java

//租客
public class Client {
   public static void main(String[] args) {
       //真實角色
       Host host = new Host();
       //代理實例的調用處理程序
       ProxyInvocationHandler pih = new ProxyInvocationHandler();
       pih.setRent(host); //將真實角色放置進去!
       Rent proxy = (Rent)pih.getProxy(); //動態生成對應的代理類!
       proxy.rent();
  }
}

核心:一個動態代理 , 一般代理某一類業務 , 一個動態代理可以代理多個類,代理的是接口!、

深化理解

我們來使用動態代理實現代理我們後面寫的UserService!

我們也可以編寫一個通用的動態代理實現的類!所有的代理對象設置為Object即可!

public class ProxyInvocationHandler implements InvocationHandler {
   private Object target;
   public void setTarget(Object target) {
       this.target = target;
  }
   //生成代理類
   public Object getProxy(){
       return Proxy.newProxyInstance(this.getClass().getClassLoader(),
               target.getClass().getInterfaces(),this);
  }
   // proxy : 代理類
   // method : 代理類的調用處理程序的方法對象.
   public Object invoke(Object proxy, Method method, Object[] args) throwsThrowable {
       log(method.getName());
       Object result = method.invoke(target, args);
       return result;
  }
   public void log(String methodName){
       System.out.println("執行瞭"+methodName+"方法");
  }
}

測試!

public class Test {
   public static void main(String[] args) {
       //真實對象
       UserServiceImpl userService = new UserServiceImpl();
       //代理對象的調用處理程序
       ProxyInvocationHandler pih = new ProxyInvocationHandler();
       pih.setTarget(userService); //設置要代理的對象
       UserService proxy = (UserService)pih.getProxy(); //動態生成代理類!
       proxy.delete();
  }
}

測試,增刪改查,查看結果!

動態代理的好處

靜態代理有的它都有,靜態代理沒有的,它也有!

  • 可以使得我們的真實角色更加純粹 . 不再去關註一些公共的事情 
  • 公共的業務由代理來完成 . 實現瞭業務的分工 
  • 公共業務發生擴展時變得更加集中和方便
  • 一個動態代理 , 一般代理某一類業務
  • 一個動態代理可以代理多個類,代理的是接口!

到此這篇關於Spring AOP底層機制之代理模式的文章就介紹到這瞭,更多相關Spring代理模式內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: