溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java靜態(tài)代理和動(dòng)態(tài)代理如何實(shí)現(xiàn)

發(fā)布時(shí)間:2022-02-24 10:30:21 來(lái)源:億速云 閱讀:148 作者:iii 欄目:開(kāi)發(fā)技術(shù)

這篇文章主要介紹“Java靜態(tài)代理和動(dòng)態(tài)代理如何實(shí)現(xiàn)”的相關(guān)知識(shí),小編通過(guò)實(shí)際案例向大家展示操作過(guò)程,操作方法簡(jiǎn)單快捷,實(shí)用性強(qiáng),希望這篇“Java靜態(tài)代理和動(dòng)態(tài)代理如何實(shí)現(xiàn)”文章能幫助大家解決問(wèn)題。

代理模式

代理模式(Proxy):為其他對(duì)象提供一個(gè)代理以控制對(duì)這個(gè)對(duì)象的訪問(wèn)。

主要解決:在直接訪問(wèn)對(duì)象時(shí)帶來(lái)的問(wèn)題,比如說(shuō):要訪問(wèn)的對(duì)象在遠(yuǎn)程的機(jī)器上。在面向?qū)ο笙到y(tǒng)中,有些對(duì)象由于某些原因(比如對(duì)象創(chuàng)建開(kāi)銷(xiāo)很大,或者某些操作需要安全控制,或者需要進(jìn)程外的訪問(wèn)),直接訪問(wèn)會(huì)給使用者或者系統(tǒng)結(jié)構(gòu)帶來(lái)很多麻煩,我們可以在訪問(wèn)此對(duì)象時(shí)加上一個(gè)對(duì)此對(duì)象的訪問(wèn)層。

代理模式的元素是:共同接口、代理對(duì)象、目標(biāo)對(duì)象。

代理模式的行為:由代理對(duì)象執(zhí)行目標(biāo)對(duì)象的方法、由代理對(duì)象擴(kuò)展目標(biāo)對(duì)象的方法。

代理模式的宏觀特性:對(duì)客戶端只暴露出接口,不暴露它以下的架構(gòu)。

好處多多:中間隔離了一層,更加符合開(kāi)閉原則

創(chuàng)建一個(gè)接口

/**
 * @Author: Promsing
 * @Date: 2021/4/3 - 8:25
 * @Description: 買(mǎi)車(chē)的接口
 * @version: 1.0
 */
public interface BuyCar {
 
 public void buyCar();
}

創(chuàng)建一個(gè)實(shí)現(xiàn)類(lèi)

/**
 * @Author: Promsing
 * @Date: 2021/4/3 - 8:25
 * @Description: 實(shí)現(xiàn)類(lèi)
 * @version: 1.0
 */
public class BuyCarImpl implements BuyCar {
 
 @Override
 public void buyCar() {
  System.out.println("我要買(mǎi)車(chē)~~~啦啦啦");
 }
}

靜態(tài)代理:

創(chuàng)建一個(gè)代理類(lèi)

 /**
 * @Author: Promsing
 * @Date: 2021/4/3 - 8:26
 * @Description: 代理類(lèi)
 * @version: 1.0
 */
public class BuyCarProxy implements BuyCar{
 private BuyCar buyCar;
 //注意事final修飾的關(guān)鍵字 不可修改
 //構(gòu)造函數(shù)注入,需要被代理的對(duì)象
 public BuyCarProxy(final BuyCar buyCar) {
  this.buyCar = buyCar;
 }
 //靜態(tài)代理- 的實(shí)現(xiàn)方式
 @Override
 public void buyCar() {
  System.out.println("不貸款,全款!買(mǎi)車(chē)前的準(zhǔn)備~~~");
  buyCar.buyCar();
  System.out.println("買(mǎi)完車(chē)了,出去浪~~~");
 }
}

客戶端調(diào)用

/**
 * @Author: Promsing
 * @Date: 2021/4/3 - 8:36
 * @Description: 客戶端調(diào)用
 * @version: 1.0
 */
public abstract class ProxyTest implements BuyCar {
  public static void main(String[] args) {
    System.out.println("-+-+-+正常調(diào)用-+-+-+");
    BuyCar car=new BuyCarImpl();
    car.buyCar();
 
    System.out.println("-+-+-+使用靜態(tài)代理-+-+-+");
    BuyCar proxy=new BuyCarProxy(car);
    proxy.buyCar();
  }
}
-+-+-+正常調(diào)用-+-+-+
我要買(mǎi)車(chē)~~~啦啦啦
 
-+-+-+使用靜態(tài)代理-+-+-+
不貸款,全款!買(mǎi)車(chē)前的準(zhǔn)備~~~
我要買(mǎi)車(chē)~~~啦啦啦
買(mǎi)完車(chē)了,出去浪~~~

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

基于接口的動(dòng)態(tài)代理類(lèi)

特點(diǎn):字節(jié)碼隨用隨創(chuàng)建,隨用隨加載

作用:在不修改源碼的基礎(chǔ)上對(duì)方法增強(qiáng)

涉及的類(lèi):JDK官方提供的Proxy

如何創(chuàng)建代理對(duì)象:使用Proxy類(lèi)中的newProxyInstance方法

創(chuàng)建代理對(duì)象的要求:被代理類(lèi)至少實(shí)現(xiàn)一個(gè)接口

newProxyInstance方法的參數(shù)

   ClassLoader:類(lèi)加載器,同于加載被代理對(duì)象字節(jié)碼

   Class[]:字節(jié)碼數(shù)組---用于讓代理對(duì)象和被代理對(duì)象擁有相同的方法

   InvocationHandler:用于提供被增強(qiáng)的代碼

/**
 * @Author: Promsing
 * @Date: 2021/4/3 - 9:09
 * @Description: 描述 形容
 * @version: 1.0
 */
public class DynamicProxy implements InvocationHandler {
  private BuyCar object;
 
  public DynamicProxy( BuyCar object) {
    this.object = object;
  }
 
  /**
   * 
   * @param proxy 代理對(duì)象的引用
   * @param method 當(dāng)前執(zhí)行的方法
   * @param args 當(dāng)前執(zhí)行方法所需的參數(shù)
   * @return 和被代理對(duì)象方法有相同的返回值
   * @throws Throwable
   */
  @Override
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    System.out.println("不貸款,全款!買(mǎi)車(chē)前的準(zhǔn)備~~~");
    Object result = method.invoke(object, args);
    System.out.println("買(mǎi)完車(chē)了,出去浪~~~");
    return result;
  }
}

客戶端

 public static void main(String[] args) {
   
    System.out.println("-+-+-+使用基于接口的代理-+-+-+");
    //方式一、如不寫(xiě)動(dòng)態(tài)代理類(lèi)DynamicProxy,可以在這里使用內(nèi)部類(lèi)
    //聲明一個(gè)final修飾的對(duì)象
    /*
    final BuyCarImpl car=new BuyCarImpl();
    BuyCar proxy=(BuyCar)Proxy.newProxyInstance(car.getClass().getClassLoader(), car.getClass().getInterfaces(), new InvocationHandler() {
      @Override
      public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("不貸款,全款!買(mǎi)車(chē)前的準(zhǔn)備~~~");
        Object result = method.invoke(car, args);
        System.out.println("買(mǎi)完車(chē)了,出去浪~~~");
        return result;
      }
    });
    proxy.buyCar();
    */
 
    //方式二、使用DynamicProxy類(lèi)
    //聲明一個(gè)final修飾的對(duì)象
    final BuyCarImpl car=new BuyCarImpl();
    BuyCar proxy=(BuyCar)Proxy.newProxyInstance(car.getClass().getClassLoader(), car.getClass().getInterfaces(),new DynamicProxy(car));
    proxy.buyCar();
  }

基于子類(lèi)的動(dòng)態(tài)代理

特點(diǎn):字節(jié)碼隨用隨創(chuàng)建,隨用隨加載

作用:在不修改源碼的基礎(chǔ)上對(duì)方法增強(qiáng)

涉及的類(lèi):第三方cglib提供的Enhancer

如何創(chuàng)建代理對(duì)象:使用Enhancer類(lèi)中create方法

創(chuàng)建代理對(duì)象的要求:被代理類(lèi)不能是最終類(lèi)

newProxyInstance方法的參數(shù)

   Class:用于被指定代理對(duì)象的字節(jié)碼

   InvocationHandler:用于提供增強(qiáng)的方法

 public static void main(String[] args) {
 
    //使用基于子類(lèi)的動(dòng)態(tài)代理
    //需要引入Jar包--cglib 本案例使用cglib3.3.0
    System.out.println("-+-+-+使用基于子類(lèi)的代理-+-+-+");
    final BuyCarImpl car=new BuyCarImpl();
     BuyCar proxy= (BuyCar)Enhancer.create(car.getClass(), new MethodInterceptor() {
      @Override
      public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("不貸款,全款!買(mǎi)車(chē)前的準(zhǔn)備~~~");
        Object result = method.invoke(car, args);
        System.out.println("買(mǎi)完車(chē)了,出去浪~~~");
        return result;
 
      }
    });
     proxy.buyCar();
  }

關(guān)于“Java靜態(tài)代理和動(dòng)態(tài)代理如何實(shí)現(xiàn)”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí),可以關(guān)注億速云行業(yè)資訊頻道,小編每天都會(huì)為大家更新不同的知識(shí)點(diǎn)。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI