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

Java動態(tài)代理和反射機制詳解

 更新時間:2019年03月19日 11:12:36   作者:MoreThinking  
這篇文章主要介紹了Java動態(tài)代理和反射機制,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

反射機制

Java語言提供的一種基礎(chǔ)功能,通過反射,我們可以操作這個類或?qū)ο?,比如獲取這個類中的方法、屬性和構(gòu)造方法等。

動態(tài)代理:分為JDK動態(tài)代理、cglib動態(tài)代理(spring中的動態(tài)代理)。

靜態(tài)代理

預先(編譯期間)確定了代理者與被代理者之間的關(guān)系,也就是說,若代理類在程序運行前就已經(jīng)存在了,這種情況就叫靜態(tài)代理

動態(tài)代理

代理類在程序運行時創(chuàng)建的代理方式。也就是說,代理類并不是在Java代碼中定義的,而是在運行期間根據(jù)我們在Java代碼中的“指示”動態(tài)生成的。

動態(tài)代理比靜態(tài)代理的優(yōu)勢在于:

動態(tài)代理可以很方便的對代理類的函數(shù)進行統(tǒng)一的處理(invoke),而不是修改每個代理類的函數(shù),更靈活和擴展。

 JDK的動態(tài)代理(依賴于接口)

  1. 在Java的動態(tài)代理機制中,有兩個重要的類或接口,一個是InvocationHandler接口,另一個是Proxy類。
  2. InvocationHandler接口是給動態(tài)代理類實現(xiàn)的,負責處理被代理對象的操作
  3. Proxy類是用來創(chuàng)建動態(tài)代理類實例對象的,只有得到這個對象,才能調(diào)用需要代理的方法。
  4. 動態(tài)代理的代理類是在靜態(tài)代理類上進行修改,將動態(tài)代理類實現(xiàn)InvocationHandler接口,重寫Invoke方法,Invoke方法通過傳入的被代理類方法和參數(shù)來執(zhí)行。

如下實例:

public interface AppService {
 void createApp(String name);
 void deleteApp(String name);
}
 
//代理類(比如微商代理)
public class AppServiceImpl implements AppService{
 
 @Override
 public void createApp(String name) {
  System.out.print("App["+name+"] has been created.");
 }
 
 @Override
 public void deleteApp(String name) {
  System.out.print("App["+name+"] has been delete.");
 }
}
 
 
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
 
public class LoggerInterceptor implements InvocationHandler {
 private Object target; //委托類(被代理類)的實例,比如廠家
 public LoggerInterceptor(Object target){ 
  this.target = target; 
 } 
 @Override
 public Object invoke(Object proxy, Method method, Object[] args)throws Throwable {
  System.out.println("Entered "+target.getClass().getName()+"-"+method.getName()+",with arguments{"+args[0]+"}"); 
  Object result = method.invoke(target, args);
  //調(diào)用目標對象的方法 (調(diào)用廠家的方法(createApp)及參數(shù)(Kevin Test))
  System.out.println("Before return:"+result); 
  return result; 
 }
 
}
import java.lang.reflect.Proxy;
 
public class test {
 
public static void main(String[] args) {
    AppService target = new AppServiceImpl();//生成目標對象 (代理類的對象)
    //接下來創(chuàng)建代理對象
    AppService proxy = (AppService) Proxy.newProxyInstance(
    target.getClass().getClassLoader(),
    target.getClass().getInterfaces(), new LoggerInterceptor(target));
    proxy.createApp("Kevin Test1");
    proxy.deleteApp("Kevin Test2");
  }
 
}
 
/**
* 1、jdk的動態(tài)代理實現(xiàn)方式是依賴于接口的,首先使用接口來定義好操作規(guī)范。
* 2、通過proxy類產(chǎn)生的代理對象調(diào)用被代理對象的操作。
* 3、而這個操作又被分發(fā)給InvocationHandler接口的invoke方法具體執(zhí)行
*
* 在java的動態(tài)代理機制中,有兩個重要的類或接口,一個是InvocationHandler接口、另一個則是 Proxy類,這個類和接口是實現(xiàn)我們動態(tài)代理所必須用到的。
InvocationHandler接口是給動態(tài)代理類實現(xiàn)的,負責處理被代理對象的操作的,而Proxy是用來創(chuàng)建動態(tài)代理類實例對象的,因為只有得到了這個對象我們才能調(diào)用那些需要代理的方法。
*
* 此方法的參數(shù)含義如下
proxy:代表動態(tài)代理對象
method:代表正在執(zhí)行的方法
args:代表當前執(zhí)行方法傳入的實參
返回值:表示當前執(zhí)行方法的返回值
*
* 如上:
* 使用了Proxy類的newProxyInstance方法生成代理對象,然后用這個對象去調(diào)用createApp()和deleteApp()方法,
* 其實系統(tǒng)會將這2個方法分發(fā)給invoke()方法區(qū)執(zhí)行。其中proxy對象的類是系統(tǒng)幫我們動態(tài)創(chuàng)建了,其實實現(xiàn)了我們的業(yè)務(wù)接口AppService
*
*/

cglib動態(tài)代理(繼承方式)

cglib動態(tài)代理中使用MethodInterceptor來實現(xiàn)動態(tài)代理類。

攔截器MethodInterceptor中就是由MethodProxy的InvokSuper方法調(diào)用代理方法的。

MethodProxy類生成代理方法和代理方法的簽名。

JDK動態(tài)代理和Cglib動態(tài)代理的區(qū)別:

  1. JDK動態(tài)代理是實現(xiàn)了被代理對象的接口,Cglib是繼承了被代理對象。
  2. Cglib因為是繼承機制,所以無法代理被final修飾的方法。
  3. JDK和Cglib都是在運行期間生產(chǎn)字節(jié)碼,JDK是直接寫class字節(jié)碼,Cglib使用ASM框架寫class字節(jié)碼;cglib代理實現(xiàn)更復雜,生成代理類比JDK效率低。
  4. JDK調(diào)用代理方法,是通過反射實現(xiàn)機制調(diào)用,cglib是通過Fashclass機制直接調(diào)用方法,效率更高。

 Fastcalss機制:

為代理類和被代理類個生成一個class,這個class會為代理類或被代理類的方法分配一個index。

這個index當做一個入?yún)ⅲ現(xiàn)ashclass就可以直接定位要調(diào)用的方法,并直接進行調(diào)用。這樣省去了反射調(diào)用,所以效率高。

以上所述是小編給大家介紹的Java動態(tài)代理和反射機制詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!

相關(guān)文章

最新評論