溫馨提示×

溫馨提示×

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

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

Java設(shè)中動態(tài)代理的原理是什么

發(fā)布時間:2021-06-11 15:11:53 來源:億速云 閱讀:160 作者:Leah 欄目:編程語言

這期內(nèi)容當中小編將會給大家?guī)碛嘘P(guān)Java設(shè)中動態(tài)代理的原理是什么,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。

一.JDK動態(tài)代理

JDK動態(tài)代理是java.lang.reflect.*包提供的方式。它必須借助一個接口才能產(chǎn)生代理對象,所以先定義接口HelloWorld:

public interface HelloWorld {
  void sayHelloWorld();
}

然后提供實現(xiàn)類HelloWorldImpl來實現(xiàn)接口:

public class HelloWorldImpl implements HelloWorld {
  @Override
  public void sayHelloWorld(){
    System.out.println("Hello World");
  }
}

這是最簡答的Java接口和實現(xiàn)類的關(guān)系,此時可以開始動態(tài)代理了。按照我們之前的分析,先要建立代理對象和真實對象的關(guān)系,然后實現(xiàn)代理邏輯,所以一共分為兩個步驟。在JDK動態(tài)代理中,代理邏輯類必須去實現(xiàn)java.lang.reflect.InvocationHandler接口,它定義了一個invoke方法,并提供接口數(shù)組用于下掛代理對象。我們自己定義一個JDK動態(tài)代理的類:

public class MyJdkProxyExample implements InvocationHandler{
  /**
   * @description 真實對象
   **/
  private Object target = null;
  /**
   * @author haozz
   * @date 2018-05-21 10:54
   * @param target 真實對象
   * @return 代理對象
   * @throws
   * @description 建立代理對象和真實對象的代理關(guān)系,并返回代理對象
   **/
  public Object getProxy(Object target) {
    this.target = target;
    return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),this);
  }
  /**
   * @author haozz
   * @date 2018-05-21 11:09
   * @param proxy 代理對象
   * @param method 當前調(diào)度方法
   * @param args 當前方法參數(shù)
   * @return 代理結(jié)果
   * @throws Throwable 異常
   * @description 代理方法邏輯
   **/
  @Override
  public Object invoke(Object proxy, Method method,Object [] args) throws Throwable{
    System.out.println("進入代理邏輯方法");
    System.out.println("在調(diào)度真實對象之前的服務(wù)");
    Object obj = method.invoke(target,args);//相當于調(diào)用sayHelloWorld方法
    System.out.println("在調(diào)度真實對象之后的服務(wù)");
    return obj;
  }
}

第一步,通過getProxy()方法建立代理對象和真實對象的關(guān)系,并返回代理對象。首先用類的屬性target保存了真實對象,然后通過Proxy.newProxyInstance()方法建立并生成代理對象,該方法中包含3個參數(shù):

  • 第1個是類加載器,這里采用了target本身的類加載器;

  • 第2個是把生成的動態(tài)代理對象下掛在哪些接口下,這個寫法就是放在target實現(xiàn)的接口下。HelloWorldImpl對象的接口顯然就是HelloWorld,代理對象可以這樣聲明:HelloWorld proxy = xxxx;;

  • 第3個是定義實現(xiàn)方法邏輯的代理類,this表示當前對象,它必須實現(xiàn)InvocationHandler接口的invoke方法,它就是代理邏輯方法的現(xiàn)實方法。

第二步,通過invoke方法實現(xiàn)代理邏輯方法。invoke方法的3個參數(shù):

  • proxy,代理對象,就是getProxy方法生成的對象;

  • method,當前調(diào)度的方法;

  • args,當前調(diào)度方法的參數(shù)。

當我們使用了代理對象調(diào)度方法后,它就會進入到invoke方法里面。

Object obj = method.invoke(target,args);

這行代碼相當于調(diào)度真實對象的方法,只是通過反射實現(xiàn)。類比前面的例子,proxy相當于商務(wù),target相當于軟件工程師,getProxy方法就是建立商務(wù)和軟件工程師之間的代理關(guān)系,invoke方法就是商務(wù)邏輯。

測試JDK動態(tài)代理:

  @Test
  public void testJdkProxy(){
    MyJdkProxyExample jdkProxy = new MyJdkProxyExample();
    //綁定關(guān)系,因為掛在接口HelloWorld下,所以聲明代理對象HelloWorld proxy
    HelloWorld proxy = (HelloWorld) jdkProxy.getProxy(new HelloWorldImpl());
    //此時HelloWorld對象已經(jīng)是一個代理對象,它會進入代理的邏輯方法invoke里
    proxy.sayHelloWorld();
  }

首先通過getProxy方法綁定了代理關(guān)系,然后在代理對象調(diào)度sayHelloWorld方法時進入了代理的邏輯,測試結(jié)果如下:

進入代理邏輯方法
在調(diào)度真實對象之前的服務(wù)
Hello World
在調(diào)度真實對象之后的服務(wù)

此時,在調(diào)度打印Hello World之前和之后都可以加入相關(guān)的邏輯,甚至可以不調(diào)度Hello World的打印。

個人小結(jié):JDK動態(tài)代理要求真實對象和代理對象之間是實現(xiàn)類和接口的關(guān)系,創(chuàng)建一個代理邏輯類實現(xiàn)InvocationHandler接口,其中g(shù)etProxy方法用于生成代理對象,然后重寫invoke方法。

二.CGLIB動態(tài)代理

JDK動態(tài)代理必須提供接口才能使用,在一些不能提供接口的環(huán)境中,只能采用其他第三方技術(shù),比如CGLIB動態(tài)代理,這里提供CGLIB動態(tài)代理的相關(guān)jar包,供學習和測試使用。它的優(yōu)勢在于不需要提供接口,只要一個非抽象類就能實現(xiàn)動態(tài)代理。

CGLIB原理:動態(tài)生成一個要代理類的子類,子類重寫要代理的類的所有不是final的方法。在子類中采用方法攔截的技術(shù)攔截所有父類方法的調(diào)用,順勢織入橫切邏輯。它比使用java反射的JDK動態(tài)代理要快。CGLIB底層:使用字節(jié)碼處理框架ASM,來轉(zhuǎn)換字節(jié)碼并生成新的類。不鼓勵直接使用ASM,因為它要求你必須對JVM內(nèi)部結(jié)構(gòu)包括class文件的格式和指令集都很熟悉。CGLIB缺點:對于final方法,無法進行代理。CGLIB廣泛地被許多AOP的框架使用,例如Spring AOP和dynaop。Hibernate使用CGLIB來代理單端single-ended(多對一和一對一)關(guān)聯(lián)。

我們以下面這個類為例:

public class ReflectServiceImpl {
  public void sayHello(String name){
    System.out.println("Hello "+name);
  }
}

它不存在實現(xiàn)任何接口,所以不能使用JDK動態(tài)代理,這里采用CGLIB動態(tài)代理技術(shù):

public class MyCglibProxyExample implements MethodInterceptor{
  /**
   * @author haozz
   * @date 2018-05-21 15:11
   * @param cls Class類
   * @return Class類的CGLIB對象
   * @throws
   * @description 生成CGLIB代理對象
   **/
  public Object getProxy(Class cls){
    //CGLIB增強類對象
    Enhancer enhancer = new Enhancer();
    //設(shè)置增強類型
    enhancer.setSuperclass(cls);
    //定義代理邏輯對象為當前對象,要求當前對象實現(xiàn)MethodInterceptor方法
    enhancer.setCallback(this);
    //生成并返回代理對象
    return enhancer.create();
  }
  /**
   * @author haozz
   * @date 2018-05-21 15:17
   * @param proxy 代理對象
   * @param method 目標方法
   * @param args 目標方法參數(shù)
   * @param methodProxy 方法代理
   * @return 代理邏輯返回
   * @throws Throwable 異常
   * @description 代理邏輯方法
   **/
  @Override
  public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy)throws Throwable{
    System.out.println("調(diào)用真實對象前");
    //CGLIB反射調(diào)用真實對象方法
    Object result = methodProxy.invokeSuper(proxy,args);
    System.out.println("調(diào)用真實對象后");
    return result;
  }
}

這里用了CGLIB的加強者Enhancer(net.sf.sglib.proxy.Enhancer),通過設(shè)置超類的方法(setSuperclass),然后通過setCallback方法設(shè)置哪個類為它的代理類。其中,參數(shù)this表示當前對象,那就要求用this這個對象實現(xiàn)接口MethodInterceptor(net.sf.sglib.proxy.MethodInterceptor)的方法intercept,然后返回代理對象。那么此時當前類的intercept方法就是其代理邏輯方法,其參數(shù)內(nèi)容見代碼注解,我們在反射真實對象方法前后進行了打印,CGLIB是通過如下代碼完成的:

Object result = methodProxy.invokeSuper(proxy,args);

測試一下CGLIB動態(tài)代理:

  @Test
  public void testCglibProxy(){
    MyCglibProxyExample cglibProxy = new MyCglibProxyExample();
    ReflectServiceImpl obj = (ReflectServiceImpl) cglibProxy.getProxy(ReflectServiceImpl.class);
    obj.sayHello("haozz");
  }

得到結(jié)果:

調(diào)用真實對象前
Hello haozz
調(diào)用真實對象后

上述就是小編為大家分享的Java設(shè)中動態(tài)代理的原理是什么了,如果剛好有類似的疑惑,不妨參照上述分析進行理解。如果想知道更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細節(jié)

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

AI