Java靜態代理和動態代理的深入講解

代理模式

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

主要解決:在直接訪問對象時帶來的問題,比如說:要訪問的對象在遠程的機器上。在面向對象系統中,有些對象由於某些原因(比如對象創建開銷很大,或者某些操作需要安全控制,或者需要進程外的訪問),直接訪問會給使用者或者系統結構帶來很多麻煩,我們可以在訪問此對象時加上一個對此對象的訪問層。

代理模式的元素是:共同接口、代理對象、目標對象。

代理模式的行為:由代理對象執行目標對象的方法、由代理對象擴展目標對象的方法。

代理模式的宏觀特性:對客戶端隻暴露出接口,不暴露它以下的架構。

好處多多:中間隔離瞭一層,更加符合開閉原則

UML圖

創建一個接口

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

創建一個實現類

/**
 * @Author: Promsing
 * @Date: 2021/4/3 - 8:25
 * @Description: 實現類
 * @version: 1.0
 */
public class BuyCarImpl implements BuyCar {
 
 @Override
 public void buyCar() {
  System.out.println("我要買車~~~啦啦啦");
 }
}

靜態代理:

創建一個代理類

 /**
 * @Author: Promsing
 * @Date: 2021/4/3 - 8:26
 * @Description: 代理類
 * @version: 1.0
 */
public class BuyCarProxy implements BuyCar{
 private BuyCar buyCar;
 //註意事final修飾的關鍵字 不可修改
 //構造函數註入,需要被代理的對象
 public BuyCarProxy(final BuyCar buyCar) {
  this.buyCar = buyCar;
 }
 //靜態代理- 的實現方式
 @Override
 public void buyCar() {
  System.out.println("不貸款,全款!買車前的準備~~~");
  buyCar.buyCar();
  System.out.println("買完車瞭,出去浪~~~");
 }
}

客戶端調用

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

 動態代理:

基於接口的動態代理類

特點:字節碼隨用隨創建,隨用隨加載

作用:在不修改源碼的基礎上對方法增強

涉及的類:JDK官方提供的Proxy

如何創建代理對象:使用Proxy類中的newProxyInstance方法

創建代理對象的要求:被代理類至少實現一個接口

newProxyInstance方法的參數

   ClassLoader:類加載器,同於加載被代理對象字節碼

   Class[]:字節碼數組—用於讓代理對象和被代理對象擁有相同的方法

   InvocationHandler:用於提供被增強的代碼

/**
 * @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 代理對象的引用
   * @param method 當前執行的方法
   * @param args 當前執行方法所需的參數
   * @return 和被代理對象方法有相同的返回值
   * @throws Throwable
   */
  @Override
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    System.out.println("不貸款,全款!買車前的準備~~~");
    Object result = method.invoke(object, args);
    System.out.println("買完車瞭,出去浪~~~");
    return result;
  }
}

客戶端

 public static void main(String[] args) {
   
    System.out.println("-+-+-+使用基於接口的代理-+-+-+");
    //方式一、如不寫動態代理類DynamicProxy,可以在這裡使用內部類
    //聲明一個final修飾的對象
    /*
    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("不貸款,全款!買車前的準備~~~");
        Object result = method.invoke(car, args);
        System.out.println("買完車瞭,出去浪~~~");
        return result;
      }
    });
    proxy.buyCar();
    */
 
    //方式二、使用DynamicProxy類
    //聲明一個final修飾的對象
    final BuyCarImpl car=new BuyCarImpl();
    BuyCar proxy=(BuyCar)Proxy.newProxyInstance(car.getClass().getClassLoader(), car.getClass().getInterfaces(),new DynamicProxy(car));
    proxy.buyCar();
  }

基於子類的動態代理

特點:字節碼隨用隨創建,隨用隨加載

作用:在不修改源碼的基礎上對方法增強

涉及的類:第三方cglib提供的Enhancer

如何創建代理對象:使用Enhancer類中create方法

創建代理對象的要求:被代理類不能是最終類

newProxyInstance方法的參數

   Class:用於被指定代理對象的字節碼

   InvocationHandler:用於提供增強的方法

 public static void main(String[] args) {
 
    //使用基於子類的動態代理
    //需要引入Jar包--cglib 本案例使用cglib3.3.0
    System.out.println("-+-+-+使用基於子類的代理-+-+-+");
    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("不貸款,全款!買車前的準備~~~");
        Object result = method.invoke(car, args);
        System.out.println("買完車瞭,出去浪~~~");
        return result;
 
      }
    });
     proxy.buyCar();
  }

總結

到此這篇關於Java靜態代理和動態代理的文章就介紹到這瞭,更多相關Java靜態代理和動態代理內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀:

    None Found