亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java 動(dòng)態(tài)代理與CGLIB詳細(xì)介紹

 更新時(shí)間:2017年02月20日 09:48:51   投稿:lqh  
這篇文章主要介紹了 Java 動(dòng)態(tài)代理與CGLIB詳細(xì)介紹的相關(guān)資料,需要的朋友可以參考下

靜態(tài)代理模式

因?yàn)樾枰獙?duì)一些函數(shù)進(jìn)行二次處理,或是某些函數(shù)不讓外界知道時(shí),可以使用代理模式,通過(guò)訪問(wèn)第三方,間接訪問(wèn)原函數(shù)的方式,達(dá)到以上目的。

interface Hosee{
  String sayhi();
}

class Hoseeimpl implements Hosee{

  @Override
  public String sayhi()
  {
    return "Welcome oschina hosee's blog";
  }

}

class HoseeProxy implements Hosee{

  Hosee h;

  public HoseeProxy(Hosee h)
  {
    this.h = h;
  }

  @Override
  public String sayhi()
  {
    System.out.println("I'm proxy!");
    return h.sayhi();
  }

}


public class StaticProxy
{

  public static void main(String[] args)
  {
    Hoseeimpl h = new Hoseeimpl();
    HoseeProxy hp = new HoseeProxy(h);
    System.out.println(hp.sayhi());
  }

}

1.1 靜態(tài)代理的弊端

如果要想為多個(gè)類(lèi)進(jìn)行代理,則需要建立多個(gè)代理類(lèi),維護(hù)難度加大。

仔細(xì)想想,為什么靜態(tài)代理會(huì)有這些問(wèn)題,是因?yàn)榇碓诰幾g期就已經(jīng)決定,如果代理哪個(gè)發(fā)生在運(yùn)行期,這些問(wèn)題解決起來(lái)就比較簡(jiǎn)單,所以動(dòng)態(tài)代理的存在就很有必要了。

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

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

interface HoseeDynamic
{
  String sayhi();
}

class HoseeDynamicimpl implements HoseeDynamic
{
  @Override
  public String sayhi()
  {
    return "Welcome oschina hosee's blog";
  }
}

class MyProxy implements InvocationHandler
{
  Object obj;
  public Object bind(Object obj)
  {
    this.obj = obj;
    return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
        .getClass().getInterfaces(), this);
  }
  @Override
  public Object invoke(Object proxy, Method method, Object[] args)
      throws Throwable
  {
    System.out.println("I'm proxy!");
    Object res = method.invoke(obj, args);
    return res;
  }
}

public class DynamicProxy
{
  public static void main(String[] args)
  {
    MyProxy myproxy = new MyProxy();
    HoseeDynamicimpl dynamicimpl = new HoseeDynamicimpl();
    HoseeDynamic proxy = (HoseeDynamic)myproxy.bind(dynamicimpl);
    System.out.println(proxy.sayhi());
  }
}

類(lèi)比靜態(tài)代理,可以發(fā)現(xiàn),代理類(lèi)不需要實(shí)現(xiàn)原接口了,而是實(shí)現(xiàn)InvocationHandler。通過(guò)

Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
        .getClass().getInterfaces(), this);

來(lái)動(dòng)態(tài)生成一個(gè)代理類(lèi),該類(lèi)的類(lèi)加載器與被代理類(lèi)相同,實(shí)現(xiàn)的接口與被代理類(lèi)相同。

通過(guò)上述方法生成的代理類(lèi)相當(dāng)于靜態(tài)代理中的代理類(lèi)。

這樣就實(shí)現(xiàn)了在運(yùn)行期才決定代理對(duì)象是怎么樣的,解決了靜態(tài)代理的弊端。

當(dāng)動(dòng)態(tài)生成的代理類(lèi)調(diào)用方法時(shí),會(huì)觸發(fā)invoke方法,在invoke方法中可以對(duì)被代理類(lèi)的方法進(jìn)行增強(qiáng)。

通過(guò)動(dòng)態(tài)代理可以很明顯的看到它的好處,在使用靜態(tài)代理時(shí),如果不同接口的某些類(lèi)想使用代理模式來(lái)實(shí)現(xiàn)相同的功能,將要實(shí)現(xiàn)多個(gè)代理類(lèi),但在動(dòng)態(tài)代理中,只需要一個(gè)代理類(lèi)就好了。

除了省去了編寫(xiě)代理類(lèi)的工作量,動(dòng)態(tài)代理實(shí)現(xiàn)了可以在原始類(lèi)和接口還未知的時(shí)候,就確定代理類(lèi)的代理行為,當(dāng)代理類(lèi)與原始類(lèi)脫離直接聯(lián)系后,就可以很靈活地重用于不同的應(yīng)用場(chǎng)景中。

2.1 動(dòng)態(tài)代理的弊端

代理類(lèi)和委托類(lèi)需要都實(shí)現(xiàn)同一個(gè)接口。也就是說(shuō)只有實(shí)現(xiàn)了某個(gè)接口的類(lèi)可以使用Java動(dòng)態(tài)代理機(jī)制。但是,事實(shí)上使用中并不是遇到的所有類(lèi)都會(huì)給你實(shí)現(xiàn)一個(gè)接口。因此,對(duì)于沒(méi)有實(shí)現(xiàn)接口的類(lèi),就不能使用該機(jī)制。

而CGLIB則可以實(shí)現(xiàn)對(duì)類(lèi)的動(dòng)態(tài)代理

2.2 回調(diào)函數(shù)原理

上文說(shuō)了,當(dāng)動(dòng)態(tài)生成的代理類(lèi)調(diào)用方法時(shí),會(huì)觸發(fā)invoke方法。

很顯然invoke方法并不是顯示調(diào)用的,它是一個(gè)回調(diào)函數(shù),那么回調(diào)函數(shù)是怎么被調(diào)用的呢?

上述動(dòng)態(tài)代理的代碼中,唯一不清晰的地方只有

Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj
        .getClass().getInterfaces(), this);

跟蹤這個(gè)方法的源碼,可以看到程序進(jìn)行了驗(yàn)證、優(yōu)化、緩存、同步、生成字節(jié)碼、顯示類(lèi)加載等操作,前面的步驟并不是我們關(guān)注的重點(diǎn),而最后它調(diào)用了

byte[] proxyClassFile = ProxyGenerator.generateProxyClass(
        proxyName, interfaces);

該方法用來(lái)完成生成字節(jié)碼的動(dòng)作,這個(gè)方法可以在運(yùn)行時(shí)產(chǎn)生一個(gè)描述代理類(lèi)的字節(jié)碼byte[]數(shù)組。

在main函數(shù)中加入

System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles","true");

加入這句代碼后再次運(yùn)行程序,磁盤(pán)中將會(huì)產(chǎn)生一個(gè)名為”$Proxy().class”的代理類(lèi)Class文件,反編譯(反編譯工具我使用的是 JD-GUI )后可以看見(jiàn)如下代碼:

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;

public final class $Proxy0 extends Proxy
 implements HoseeDynamic
{
 private static Method m1;
 private static Method m3;
 private static Method m0;
 private static Method m2;

 public $Proxy0(InvocationHandler paramInvocationHandler)
  throws 
 {
  super(paramInvocationHandler);
 }

 public final boolean equals(Object paramObject)
  throws 
 {
  try
  {
   return ((Boolean)this.h.invoke(this, m1, new Object[] { paramObject })).booleanValue();
  }
  catch (Error|RuntimeException localError)
  {
   throw localError;
  }
  catch (Throwable localThrowable)
  {
   throw new UndeclaredThrowableException(localThrowable);
  }
 }

 public final String sayhi()
  throws 
 {
  try
  {
   return (String)this.h.invoke(this, m3, null);
  }
  catch (Error|RuntimeException localError)
  {
   throw localError;
  }
  catch (Throwable localThrowable)
  {
   throw new UndeclaredThrowableException(localThrowable);
  }
 }

 public final int hashCode()
  throws 
 {
  try
  {
   return ((Integer)this.h.invoke(this, m0, null)).intValue();
  }
  catch (Error|RuntimeException localError)
  {
   throw localError;
  }
  catch (Throwable localThrowable)
  {
   throw new UndeclaredThrowableException(localThrowable);
  }
 }

 public final String toString()
  throws 
 {
  try
  {
   return (String)this.h.invoke(this, m2, null);
  }
  catch (Error|RuntimeException localError)
  {
   throw localError;
  }
  catch (Throwable localThrowable)
  {
   throw new UndeclaredThrowableException(localThrowable);
  }
 }

 static
 {
  try
  {
   m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[] { Class.forName("java.lang.Object") });
   m3 = Class.forName("HoseeDynamic").getMethod("sayhi", new Class[0]);
   m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
   m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
   return;
  }
  catch (NoSuchMethodException localNoSuchMethodException)
  {
   throw new NoSuchMethodError(localNoSuchMethodException.getMessage());
  }
  catch (ClassNotFoundException localClassNotFoundException)
  {
   throw new NoClassDefFoundError(localClassNotFoundException.getMessage());
  }
 }
}

動(dòng)態(tài)代理類(lèi)不僅代理了顯示定義的接口中的方法,而且還代理了java的根類(lèi)Object中的繼承而來(lái)的equals()、hashcode()、toString()這三個(gè)方法,并且僅此三個(gè)方法。

可以在上述代碼中看到,無(wú)論調(diào)用哪個(gè)方法,都會(huì)調(diào)用到InvocationHandler的invoke方法,只是參數(shù)不同。

2.3 動(dòng)態(tài)代理與靜態(tài)代理的區(qū)別

Proxy類(lèi)的代碼被固定下來(lái),不會(huì)因?yàn)闃I(yè)務(wù)的逐漸龐大而龐大;

可以實(shí)現(xiàn)AOP編程,這是靜態(tài)代理無(wú)法實(shí)現(xiàn)的;

解耦,如果用在web業(yè)務(wù)下,可以實(shí)現(xiàn)數(shù)據(jù)層和業(yè)務(wù)層的分離。

動(dòng)態(tài)代理的優(yōu)勢(shì)就是實(shí)現(xiàn)無(wú)侵入式的代碼擴(kuò)展。 靜態(tài)代理這個(gè)模式本身有個(gè)大問(wèn)題,如果類(lèi)方法數(shù)量越來(lái)越多的時(shí)候,代理類(lèi)的代碼量是十分龐大的。所以引入動(dòng)態(tài)代理來(lái)解決此類(lèi)問(wèn)題

3. CGLIB

cglib是針對(duì)類(lèi)來(lái)實(shí)現(xiàn)代理的,他的原理是對(duì)指定的目標(biāo)類(lèi)生成一個(gè)子類(lèi),并覆蓋其中方法實(shí)現(xiàn)增強(qiáng),但因?yàn)椴捎玫氖抢^承,所以不能對(duì)final修飾的類(lèi)進(jìn)行代理。

import java.lang.reflect.Method;

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

class CGlibHosee
{
  public String sayhi()
  {
    return "Welcome oschina hosee's blog";
  }
}

class CGlibHoseeProxy
{
  Object obj;

  public Object bind(final Object target)
  {
    this.obj = target;
    Enhancer enhancer = new Enhancer();
    enhancer.setSuperclass(obj.getClass());
    enhancer.setCallback(new MethodInterceptor()
    {
      @Override
      public Object intercept(Object obj, Method method, Object[] args,
          MethodProxy proxy) throws Throwable
      {
        System.out.println("I'm proxy!");
        Object res = method.invoke(target, args);
        return res;
      }
    });
    return enhancer.create();
  }

}

public class CGlibProxy
{
  public static void main(String[] args)
  {
    CGlibHosee cGlibHosee = new CGlibHosee();
    CGlibHoseeProxy cGlibHoseeProxy = new CGlibHoseeProxy();
    CGlibHosee proxy = (CGlibHosee) cGlibHoseeProxy.bind(cGlibHosee);
    System.out.println(proxy.sayhi());
  }
}

cglib需要指定父類(lèi)和回調(diào)方法。當(dāng)然cglib也可以與Java動(dòng)態(tài)代理一樣面向接口,因?yàn)楸举|(zhì)是繼承。

感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!

相關(guān)文章

  • 詳解Spring中Bean的加載的方法

    詳解Spring中Bean的加載的方法

    本篇文章主要介紹了Spring中Bean的加載的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-04-04
  • Java下載文件的4種方式總結(jié)

    Java下載文件的4種方式總結(jié)

    這篇文章主要給大家總結(jié)介紹了關(guān)于Java下載文件的4種方式,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-01-01
  • java根據(jù)負(fù)載自動(dòng)抓取jstack?dump詳情

    java根據(jù)負(fù)載自動(dòng)抓取jstack?dump詳情

    這篇文章主要介紹了java根據(jù)負(fù)載自動(dòng)抓取jstack?dump詳情,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下
    2022-09-09
  • JAVA中常見(jiàn)異常類(lèi)

    JAVA中常見(jiàn)異常類(lèi)

    本文主要介紹了JAVA中的常見(jiàn)異常類(lèi)。具有很好的參考價(jià)值,下面跟著小編一起來(lái)看下吧
    2017-01-01
  • windows系統(tǒng)使用mvn命令打包并指定jdk路徑方式

    windows系統(tǒng)使用mvn命令打包并指定jdk路徑方式

    這篇文章主要介紹了windows系統(tǒng)使用mvn命令打包并指定jdk路徑方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-04-04
  • jdbc和mybatis的流式查詢(xún)使用方法

    jdbc和mybatis的流式查詢(xún)使用方法

    有些時(shí)候我們所需要查詢(xún)的數(shù)據(jù)量比較大,但是jvm內(nèi)存又是有限制的,數(shù)據(jù)量過(guò)大會(huì)導(dǎo)致內(nèi)存溢出。這個(gè)時(shí)候就可以使用流式查詢(xún),本文就主要介紹了jdbc和mybatis的流式查詢(xún),感興趣的可以了解一下
    2021-11-11
  • add方法理解ArrayList的擴(kuò)容機(jī)制

    add方法理解ArrayList的擴(kuò)容機(jī)制

    這篇文章主要為大家介紹了add方法理解ArrayList的擴(kuò)容機(jī)制示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-03-03
  • java運(yùn)行錯(cuò)誤A JNI error的解決方案

    java運(yùn)行錯(cuò)誤A JNI error的解決方案

    這篇文章主要介紹了java運(yùn)行錯(cuò)誤A JNI error的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-08-08
  • jpa實(shí)現(xiàn)只查詢(xún)指定的字段

    jpa實(shí)現(xiàn)只查詢(xún)指定的字段

    這篇文章主要介紹了jpa實(shí)現(xiàn)只查詢(xún)指定的字段,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-12-12
  • SpringBoot如何統(tǒng)一JSON信息返回

    SpringBoot如何統(tǒng)一JSON信息返回

    這篇文章主要介紹了SpringBoot如何統(tǒng)一JSON信息返回問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-08-08

最新評(píng)論