欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring?AOP底層機(jī)制之代理模式

 更新時(shí)間:2022年08月08日 16:45:43   作者:進(jìn)擊攻城獅  
如何實(shí)現(xiàn)在不修改源碼的基礎(chǔ)上實(shí)現(xiàn)代碼功能的增強(qiáng)呢?spring為我們提供了代理模式。所謂的代理模式通俗來說就是一個(gè)中介,它給某一個(gè)對象提供一個(gè)代理對象,并由代理對象控制原對象的引用,從而實(shí)現(xiàn)在不修改源碼的基礎(chǔ)上實(shí)現(xiàn)代碼功能的增強(qiáng)

代理模式

為什么要學(xué)習(xí)代理模式,因?yàn)锳OP的底層機(jī)制就是動(dòng)態(tài)代理!

代理模式:

  • 靜態(tài)代理
  • 動(dòng)態(tài)代理

學(xué)習(xí)aop之前 , 我們要先了解一下代理模式!

靜態(tài)代理

靜態(tài)代理角色分析

  • 抽象角色 : 一般使用接口或者抽象類來實(shí)現(xiàn)
  • 真實(shí)角色 : 被代理的角色
  • 代理角色 : 代理真實(shí)角色 ; 代理真實(shí)角色后 , 一般會做一些附屬的操作 .
  • 客戶 : 使用代理角色來進(jìn)行一些操作 .

代碼實(shí)現(xiàn)

Rent . java 即抽象角色

public interface Rent {
    public void rent();
}

Host . java 即真實(shí)角色

//真實(shí)角色: 房東,房東要出租房子
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("帶房客看房");
  }
   //收中介費(fèi)
   public void fare(){
       System.out.println("收中介費(fèi)");
  }
}

Client . java 即客戶

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

分析:在這個(gè)過程中,你直接接觸的就是中介,就如同現(xiàn)實(shí)生活中的樣子,你看不到房東,但是你依舊租到了房東的房子通過代理,這就是所謂的代理模式,程序源自于生活,所以學(xué)編程的人,一般能夠更加抽象的看待生活中發(fā)生的事情。

靜態(tài)代理的好處:

  • 可以使得我們的真實(shí)角色更加純粹 . 不再去關(guān)注一些公共的事情 .
  • 公共的業(yè)務(wù)由代理來完成 . 實(shí)現(xiàn)了業(yè)務(wù)的分工 ,
  • 公共業(yè)務(wù)發(fā)生擴(kuò)展時(shí)變得更加集中和方便 .

缺點(diǎn) :

類多了 , 多了代理類 , 工作量變大了 . 開發(fā)效率降低 .

我們想要靜態(tài)代理的好處,又不想要靜態(tài)代理的缺點(diǎn),所以 , 就有了動(dòng)態(tài)代理 !

靜態(tài)代理再理解

同學(xué)們練習(xí)完畢后,我們再來舉一個(gè)例子,鞏固大家的學(xué)習(xí)!

練習(xí)步驟:

1、創(chuàng)建一個(gè)抽象角色,比如咋們平時(shí)做的用戶業(yè)務(wù),抽象起來就是增刪改查!

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

2、我們需要一個(gè)真實(shí)對象來完成這些增刪改查操作

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

3、需求來了,現(xiàn)在我們需要增加一個(gè)日志功能,怎么實(shí)現(xiàn)!

  • 思路1 :在實(shí)現(xiàn)類上增加代碼 【麻煩!】
  • 思路2:使用代理來做,能夠不改變原來的業(yè)務(wù)情況下,實(shí)現(xiàn)此功能就是最好的了!

4、設(shè)置一個(gè)代理類來處理日志!代理角色

//代理角色,在這里面增加日志的實(shí)現(xiàn)
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("執(zhí)行了"+msg+"方法");
  }
}

5、測試訪問類:

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

OK,到了現(xiàn)在代理模式大家應(yīng)該都沒有什么問題了,重點(diǎn)大家需要理解其中的思想;

我們在不改變原來的代碼的情況下,實(shí)現(xiàn)了對原有功能的增強(qiáng),這是AOP中最核心的思想

聊聊AOP:縱向開發(fā),橫向開發(fā)

動(dòng)態(tài)代理

動(dòng)態(tài)代理的角色和靜態(tài)代理的一樣 .

動(dòng)態(tài)代理的代理類是動(dòng)態(tài)生成的 . 靜態(tài)代理的代理類是我們提前寫好的

動(dòng)態(tài)代理分為兩類 : 一類是基于接口動(dòng)態(tài)代理 , 一類是基于類的動(dòng)態(tài)代理

  • 基于接口的動(dòng)態(tài)代理----JDK動(dòng)態(tài)代理
  • 基于類的動(dòng)態(tài)代理–cglib
  • 現(xiàn)在用的比較多的是 javasist 來生成動(dòng)態(tài)代理 . 百度一下javasist
  • 我們這里使用JDK的原生代碼來實(shí)現(xiàn),其余的道理都是一樣的!

JDK的動(dòng)態(tài)代理需要了解兩個(gè)類

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

【InvocationHandler:調(diào)用處理程序】

Object invoke(Object proxy, 方法 method, Object[] args);
//參數(shù)
//proxy - 調(diào)用該方法的代理實(shí)例
//method -所述方法對應(yīng)于調(diào)用代理實(shí)例上的接口方法的實(shí)例。方法對象的聲明類將是該方法聲明的接口,它可以是代理類繼承該方法的代理接口的超級接口。
//args -包含的方法調(diào)用傳遞代理實(shí)例的參數(shù)值的對象的陣列,或null如果接口方法沒有參數(shù)。原始類型的參數(shù)包含在適當(dāng)?shù)脑及b器類的實(shí)例中,例如java.lang.Integer或java.lang.Boolean 。

【Proxy : 代理】

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

代碼實(shí)現(xiàn)

抽象角色和真實(shí)角色和之前的一樣!

Rent . java 即抽象角色

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

Host . java 即真實(shí)角色

//真實(shí)角色: 房東,房東要出租房子
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;
  }
   //生成代理類,重點(diǎn)是第二個(gè)參數(shù),獲取要代理的抽象角色!之前都是一個(gè)角色,現(xiàn)在可以代理一類角色
   public Object getProxy(){
       return Proxy.newProxyInstance(this.getClass().getClassLoader(),
               rent.getClass().getInterfaces(),this);
  }
   // proxy : 代理類 method : 代理類的調(diào)用處理程序的方法對象.
   // 處理代理實(shí)例上的方法調(diào)用并返回結(jié)果
   @Override
   public Object invoke(Object proxy, Method method, Object[] args) throwsThrowable {
       seeHouse();
       //核心:本質(zhì)利用反射實(shí)現(xiàn)!
       Object result = method.invoke(rent, args);
       fare();
       return result;
  }
   //看房
   public void seeHouse(){
       System.out.println("帶房客看房");
  }
   //收中介費(fèi)
   public void fare(){
       System.out.println("收中介費(fèi)");
  }
}

Client . java

//租客
public class Client {
   public static void main(String[] args) {
       //真實(shí)角色
       Host host = new Host();
       //代理實(shí)例的調(diào)用處理程序
       ProxyInvocationHandler pih = new ProxyInvocationHandler();
       pih.setRent(host); //將真實(shí)角色放置進(jìn)去!
       Rent proxy = (Rent)pih.getProxy(); //動(dòng)態(tài)生成對應(yīng)的代理類!
       proxy.rent();
  }
}

核心:一個(gè)動(dòng)態(tài)代理 , 一般代理某一類業(yè)務(wù) , 一個(gè)動(dòng)態(tài)代理可以代理多個(gè)類,代理的是接口!、

深化理解

我們來使用動(dòng)態(tài)代理實(shí)現(xiàn)代理我們后面寫的UserService!

我們也可以編寫一個(gè)通用的動(dòng)態(tài)代理實(shí)現(xiàn)的類!所有的代理對象設(shè)置為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 : 代理類的調(diào)用處理程序的方法對象.
   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("執(zhí)行了"+methodName+"方法");
  }
}

測試!

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

測試,增刪改查,查看結(jié)果!

動(dòng)態(tài)代理的好處

靜態(tài)代理有的它都有,靜態(tài)代理沒有的,它也有!

  • 可以使得我們的真實(shí)角色更加純粹 . 不再去關(guān)注一些公共的事情 
  • 公共的業(yè)務(wù)由代理來完成 . 實(shí)現(xiàn)了業(yè)務(wù)的分工 
  • 公共業(yè)務(wù)發(fā)生擴(kuò)展時(shí)變得更加集中和方便
  • 一個(gè)動(dòng)態(tài)代理 , 一般代理某一類業(yè)務(wù)
  • 一個(gè)動(dòng)態(tài)代理可以代理多個(gè)類,代理的是接口!

到此這篇關(guān)于Spring AOP底層機(jī)制之代理模式的文章就介紹到這了,更多相關(guān)Spring代理模式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java如何實(shí)現(xiàn)通過鍵盤輸入一個(gè)數(shù)組

    Java如何實(shí)現(xiàn)通過鍵盤輸入一個(gè)數(shù)組

    這篇文章主要介紹了Java實(shí)現(xiàn)通過鍵盤輸入一個(gè)數(shù)組,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-02-02
  • Mybatis詳解在注解sql時(shí)報(bào)錯(cuò)的解決方法

    Mybatis詳解在注解sql時(shí)報(bào)錯(cuò)的解決方法

    MyBatis-Plus 是一個(gè) Mybatis 增強(qiáng)版工具,在 MyBatis 上擴(kuò)充了其他功能沒有改變其基本功能,為了簡化開發(fā)提交效率而存在,本篇文章帶你看看在注解sql時(shí)所報(bào)出的錯(cuò)誤解決
    2022-03-03
  • Springboot集成定時(shí)器和多線程異步處理操作

    Springboot集成定時(shí)器和多線程異步處理操作

    這篇文章主要介紹了Springboot集成定時(shí)器和多線程異步處理操作,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-09-09
  • SpringBoot文件上傳的原理解析

    SpringBoot文件上傳的原理解析

    這篇文章主要介紹了SpringBoot文件上傳的原理解析,SpringBoot 文件上傳是一種方便快捷的方式,可以將文件上傳到服務(wù)器,通過使用SpringBoot的文件上傳功能,可以輕松地實(shí)現(xiàn)文件上傳功能,需要的朋友可以參考下
    2023-10-10
  • SpringBoot整合RestTemplate用法的實(shí)現(xiàn)

    SpringBoot整合RestTemplate用法的實(shí)現(xiàn)

    本篇主要介紹了RestTemplate中的GET,POST,PUT,DELETE、文件上傳和文件下載6大常用的功能,具有一定的參考價(jià)值,感興趣的可以了解一下
    2023-08-08
  • POST方法給@RequestBody傳參數(shù)失敗的解決及原因分析

    POST方法給@RequestBody傳參數(shù)失敗的解決及原因分析

    這篇文章主要介紹了POST方法給@RequestBody傳參數(shù)失敗的解決及原因分析,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • 值得收藏的2017年Java開發(fā)崗位面試題

    值得收藏的2017年Java開發(fā)崗位面試題

    這篇文章主要為大家推薦一份值得收藏的2017年Java開發(fā)崗位面試題,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-08-08
  • Spring Boot中實(shí)現(xiàn)定時(shí)任務(wù)應(yīng)用實(shí)踐

    Spring Boot中實(shí)現(xiàn)定時(shí)任務(wù)應(yīng)用實(shí)踐

    定時(shí)任務(wù)一般是項(xiàng)目中都需要用到的,可以用于定時(shí)處理一些特殊的任務(wù)。下面這篇文章主要給大家介紹了關(guān)于Spring Boot中實(shí)現(xiàn)定時(shí)任務(wù)應(yīng)用實(shí)踐的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下。
    2018-05-05
  • Java foreach循環(huán)的使用方法詳解

    Java foreach循環(huán)的使用方法詳解

    Java SE5引入了一種更加簡潔的for語法用于數(shù)組和容器,即foreach語法,表示不必創(chuàng)建int變量去對由訪問項(xiàng)構(gòu)成的序列進(jìn)行計(jì)數(shù),foreach將自動(dòng)產(chǎn)生每一項(xiàng),這種循環(huán)方式在我們后來遍歷集合時(shí)很常用,所以也有必要來學(xué)習(xí)一下,需要的朋友可以參考下
    2023-05-05
  • Java如何獲取主機(jī)的基本信息詳解

    Java如何獲取主機(jī)的基本信息詳解

    最近遇到一個(gè)工作需求,上網(wǎng)查了一下怎樣在Java中獲取本機(jī)的ip和主機(jī)名,所以下面這篇文章主要給大家介紹了關(guān)于Java如何獲取主機(jī)的基本信息,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2021-12-12

最新評論