日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
Java動(dòng)態(tài)代理的實(shí)現(xiàn)機(jī)制

一、概述

十載專注成都網(wǎng)站制作,成都定制網(wǎng)頁設(shè)計(jì),個(gè)人網(wǎng)站制作服務(wù),為大家分享網(wǎng)站制作知識(shí)、方案,網(wǎng)站設(shè)計(jì)流程、步驟,成功服務(wù)上千家企業(yè)。為您提供網(wǎng)站建設(shè),網(wǎng)站制作,網(wǎng)頁設(shè)計(jì)及定制高端網(wǎng)站建設(shè)服務(wù),專注于成都定制網(wǎng)頁設(shè)計(jì),高端網(wǎng)頁制作,對(duì)搬家公司等多個(gè)領(lǐng)域,擁有豐富的網(wǎng)站營銷經(jīng)驗(yàn)。

  代理是一種設(shè)計(jì)模式,其目的是為其他對(duì)象提供一個(gè)代理以控制對(duì)某個(gè)對(duì)象的訪問,代理類負(fù)責(zé)為委托類預(yù)處理消息,過濾消息并轉(zhuǎn)發(fā)消息以及進(jìn)行消息被委托類執(zhí)行后的后續(xù)處理。為了保持行為的一致性,代理類和委托類通常會(huì)實(shí)現(xiàn)相同的接口。

  按照代理的創(chuàng)建時(shí)期,代理類可分為兩種:

  • 靜態(tài)代理:由程序員創(chuàng)建代理類或特定工具自動(dòng)生成源代碼再對(duì)其編譯,也就是說在程序運(yùn)行前代理類的.class文件就已經(jīng)存在。

  • 動(dòng)態(tài)代理:在程序運(yùn)行時(shí)運(yùn)用反射機(jī)制動(dòng)態(tài)創(chuàng)建生成。

  下面在將動(dòng)態(tài)代理的實(shí)現(xiàn)機(jī)制之前先簡單介紹一下靜態(tài)代理。

二、靜態(tài)代理

  上面說過,代理類和委托類一般都要實(shí)現(xiàn)相同的接口,下面先定義這個(gè)接口:

  
 
  1. public interface Service
  2. {    
  3.     public void add();
  4. }

  委托類作為接口的一種實(shí)現(xiàn),定義如下:

 
 
  1. public class ServiceImpl implements Service
  2. {
  3.     public void add()
  4.     {
  5.         System.out.println("添加用戶!");
  6.         
  7.     }
  8. }

  假如我們要對(duì)委托類加一些日志的操作,代理類可做如下定義:

 
 
  1. public class ServiceProxy implements Service
  2. {
  3.     private Service service;
  4.     public ServiceProxy(Service service)
  5.     {
  6.         super();
  7.         this.service = service;
  8.     }
  9.     public void add()
  10.     {
  11.         System.out.println("服務(wù)開始");
  12.         service.add();
  13.         System.out.println("服務(wù)結(jié)束");
  14.     }
  15. }

  編寫測試類:

 
 
  1. public class TestMain
  2. {
  3.     public static void main(String[] args)
  4.     {
  5.         Service serviceImpl=new ServiceImpl();
  6.         Service proxy=new ServiceProxy(serviceImpl);
  7.         proxy.add();
  8.     }
  9. }

  運(yùn)行測試程序,結(jié)果如下圖:

  從上面的代碼可以看到,靜態(tài)代理類只能為特定的接口服務(wù),如果要服務(wù)多類型的對(duì)象,就要為每一種對(duì)象進(jìn)行代理。我們就會(huì)想是否可以通過一個(gè)代理類完成全部的代理功能,于是引入的動(dòng)態(tài)代理的概念。

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

  Java的動(dòng)態(tài)代理主要涉及兩個(gè)類,Proxy和InvocationHandler。

  Proxy:提供了一組靜態(tài)方法來為一組接口動(dòng)態(tài)地生成代理類及其對(duì)象。

 
 
  1. // 方法 1: 該方法用于獲取指定代理對(duì)象所關(guān)聯(lián)的調(diào)用處理器
  2. static InvocationHandler getInvocationHandler(Object proxy) 
  3. // 方法 2:該方法用于獲取關(guān)聯(lián)于指定類裝載器和一組接口的動(dòng)態(tài)代理類的類對(duì)象
  4. static Class getProxyClass(ClassLoader loader, Class[] interfaces) 
  5. // 方法 3:該方法用于判斷指定類對(duì)象是否是一個(gè)動(dòng)態(tài)代理類
  6. static boolean isProxyClass(Class cl) 
  7. // 方法 4:該方法用于為指定類裝載器、一組接口及調(diào)用處理器生成動(dòng)態(tài)代理類實(shí)例
  8. static Object newProxyInstance(ClassLoader loader, Class[] interfaces,InvocationHandler h)

  InvocationHandler:它是調(diào)用處理器接口,自定義了一個(gè)invok方法,用于集中處理在動(dòng)態(tài)代理類對(duì)象上的方法調(diào)用,通常在該方法中實(shí)現(xiàn)對(duì)委托類的代理訪問

  
 
  1. // 該方法負(fù)責(zé)集中處理動(dòng)態(tài)代理類上的所有方法調(diào)用。***個(gè)參數(shù)既是代理類實(shí)例,第二個(gè)參數(shù)是被調(diào)用的方法對(duì)象// 第三個(gè)方法是調(diào)用參數(shù)。調(diào)用處理器根據(jù)這三個(gè)參數(shù)進(jìn)行預(yù)處理或分派到委托類實(shí)例上發(fā)射執(zhí)行
  2. Object invoke(Object proxy, Method method, Object[] args)

  實(shí)現(xiàn)Java的動(dòng)態(tài)代理,具體有以下四個(gè)步驟:

  1. 通過實(shí)現(xiàn)InvocationHandler接口創(chuàng)建自己的調(diào)用處理器

  2. 通過為Proxy類指定ClassLoader對(duì)象和一組interface來創(chuàng)建動(dòng)態(tài)代理類

  3. 通過反射機(jī)制獲得動(dòng)態(tài)代理類的構(gòu)造函數(shù),其***參數(shù)類型是調(diào)用處理器類接口類型

  4. 通過構(gòu)造函數(shù)創(chuàng)建動(dòng)態(tài)代理類實(shí)例,構(gòu)造時(shí)調(diào)用處理器對(duì)象作為參數(shù)被傳入

  下面根據(jù)上述的四個(gè)步驟來實(shí)現(xiàn)自己的動(dòng)態(tài)代理的示例:

  接口和接口的實(shí)現(xiàn)類(即委托類)跟上面靜態(tài)代理的代碼一樣,這里我們來實(shí)現(xiàn)InvocationHandler接口創(chuàng)建自己的調(diào)用處理器

 
 
  1. public class ServiceHandle implements InvocationHandler
  2. {
  3.     private Object s;
  4.     
  5.     public ServiceHandle(Object s)
  6.     {
  7.         this.s = s;
  8.     }
  9.     public Object invoke(Object proxy, Method method, Object[] args)
  10.             throws Throwable
  11.     {
  12.         System.out.println("服務(wù)開始");
  13.         //invoke表示對(duì)帶有指定參數(shù)的指定對(duì)象調(diào)用由此 Method 對(duì)象表示的底層方法
  14.         Object result=method.invoke(s, args);
  15.         System.out.println("服務(wù)結(jié)束");
  16.         return result;
  17.     }
  18. }

  編寫測試類:

 
 
  1. public class TestMain
  2. {
  3.     public static void main(String[] args)
  4.     {
  5.         Service service=new ServiceImpl();
  6.         InvocationHandler handler=new ServiceHandle(service);
  7.         Service s=(Service) Proxy.newProxyInstance(service.getClass().getClassLoader(), service.getClass().getInterfaces(), handler);
  8.         s.add();
  9.     }
  10. }

  運(yùn)行測試程序,結(jié)果同靜態(tài)代理。我們可以看到上述代碼并沒有我們之前說的步驟2和3,這是因?yàn)镻rox的靜態(tài)方法newProxyInstance已經(jīng)為我們封裝了這兩個(gè)步驟。具體的內(nèi)部實(shí)現(xiàn)如下:

 
 
  1. // 通過 Proxy 為包括 Interface 接口在內(nèi)的一組接口動(dòng)態(tài)創(chuàng)建代理類的類對(duì)象
  2. Class clazz = Proxy.getProxyClass(classLoader, new Class[] { Interface.class, ... }); 
  3. // 通過反射從生成的類對(duì)象獲得構(gòu)造函數(shù)對(duì)象
  4. Constructor constructor = clazz.getConstructor(new Class[] { InvocationHandler.class }); 
  5. // 通過構(gòu)造函數(shù)對(duì)象創(chuàng)建動(dòng)態(tài)代理類實(shí)例
  6. Interface Proxy = (Interface)constructor.newInstance(new Object[] { handler });

  newProxyInstance函數(shù)的內(nèi)部實(shí)現(xiàn)為:

 
 
  1. public static Object newProxyInstance(ClassLoader loader, Class[] interfaces,InvocationHandler h)throws IllegalArgumentException
  2.   {
  3.              //檢查h不為空,否則拋異常
  4.             Objects.requireNonNull(h);
  5.             //獲得與制定類裝載器和一組接口相關(guān)的代理類類型對(duì)象
  6.             final Class[] intfs = interfaces.clone();
  7.             
  8.             //檢查接口類對(duì)象是否對(duì)類裝載器可見并且與類裝載器所能識(shí)別的接口類對(duì)象是完全相同的
  9.             final SecurityManager sm = System.getSecurityManager();    
  10.             if (sm != null) 
  11.             {
  12.                 checkProxyAccess(Reflection.getCallerClass(), loader, intfs);
  13.             }
  14.             //獲得與制定類裝載器和一組接口相關(guān)的代理類類型對(duì)象
  15.             Class cl = getProxyClass0(loader, intfs);
  16.             try
  17.             {
  18.                 if (sm != null) 
  19.                 {
  20.                     checkNewProxyPermission(Reflection.getCallerClass(), cl);
  21.                 }
  22.                 // 通過反射獲取構(gòu)造函數(shù)對(duì)象并生成代理類實(shí)例
  23.                 final Constructor cons = cl.getConstructor(constructorParams);
  24.                 final InvocationHandler ih = h;
  25.                 if (!Modifier.isPublic(cl.getModifiers())) 
  26.                 {
  27.                     AccessController.doPrivileged(new PrivilegedAction() 
  28.                     {
  29.                         public Void run() 
  30.                         {
  31.                         cons.setAccessible(true);
  32.                         return null;
  33.                         }
  34.                     });
  35.                 }
  36.                 return cons.newInstance(new Object[]{h});
  37.             } 
  38.             catch (IllegalAccessException|InstantiationException e)
  39.             {
  40.                 throw new InternalError(e.toString(), e);
  41.             }
  42.             catch (InvocationTargetException e) 
  43.             {
  44.                 Throwable t = e.getCause();
  45.                 if (t instanceof RuntimeException) 
  46.                 {
  47.                     throw (RuntimeException) t;
  48.                 }
  49.                 else 
  50.                 {
  51.                     throw new InternalError(t.toString(), t);
  52.                 }
  53.             } 
  54.             catch (NoSuchMethodException e) 
  55.             {
  56.                 throw new InternalError(e.toString(), e);
  57.            }
  58.  }

四、模擬實(shí)現(xiàn)Proxy類

  根據(jù)上面的原理介紹,我們可以自己模擬實(shí)現(xiàn)Proxy類:

 
 
  1. public class Proxy
  2. {
  3.     public static Object newProxyInstance(Class inface,InvocationHandle h) throws Exception
  4.     {
  5.         String rt="\r\n";
  6.         String methodStr="";
  7.         Method[] methods=inface.getMethods();
  8.         for(Method m:methods)
  9.         {
  10.             methodStr+="@Override"+rt+
  11.                      "public void "+m.getName()+"()"+rt+"{" + rt +
  12.                     "   try {"+rt+
  13.                     "  Method md="+inface.getName()+".class.getMethod(\""+m.getName()+"\");"+rt+
  14.                         "h.invoke(this,md);"+rt+
  15.                     "   } catch(Exception e){e.printStackTrace();}"+rt+
  16.                     
  17.                     "}";
  18.         }
  19.         String src="package test;"+rt+
  20.                 "import java.lang.reflect.Method;"+rt+
  21.                 "public class ServiceImpl2 implements "+inface.getName()+ rt+
  22.                 "{"+rt+
  23.                     "public ServiceImpl2(InvocationHandle h)"+rt+
  24.                     "{"+rt+
  25.                         "this.h = h;"+rt+
  26.                     "}"+rt+
  27.                     " test.InvocationHandle h;"+rt+
  28.                     methodStr+
  29.                 "}";
  30.         String fileName="d:/src/test/ServiceImpl2.java";
  31.         //compile
  32.         compile(src, fileName);
  33.         //load into memory and create instance
  34.         Object m = loadMemory(h);
  35.         
  36.         return m;
  37.     }
  38.     private static void compile(String src, String fileName) throws IOException
  39.     {
  40.         File f=new File(fileName);
  41.         FileWriter fileWriter=new FileWriter(f);
  42.         fileWriter.write(src);
  43.         fileWriter.flush();
  44.         fileWriter.close();
  45.         //獲取此平臺(tái)提供的Java編譯器
  46.         JavaCompiler compiler=ToolProvider.getSystemJavaCompiler();
  47.         //獲取一個(gè)標(biāo)準(zhǔn)文件管理器實(shí)現(xiàn)的新實(shí)例
  48.         StandardJavaFileManager fileManager=compiler.getStandardFileManager(null,null, null);
  49.         //獲取表示給定文件的文件對(duì)象
  50.         Iterable units=fileManager.getJavaFileObjects(fileName);
  51.         //使用給定組件和參數(shù)創(chuàng)建編譯任務(wù)的 future
  52.         CompilationTask t=compiler.getTask(null, fileManager, null, null, null, units);
  53.         //執(zhí)行此編譯任務(wù)
  54.         t.call();    
  55.         fileManager.close();
  56.     }
  57.     private static Object loadMemory(InvocationHandle h)
  58.             throws MalformedURLException, ClassNotFoundException,
  59.             NoSuchMethodException, InstantiationException,
  60.             IllegalAccessException, InvocationTargetException
  61.     {
  62.         URL[] urls=new URL[] {new URL("file:/"+"d:/src/")};
  63.         //從路徑d:/src/加載類和資源
  64.         URLClassLoader ul=new URLClassLoader(urls);
  65.         Class c=ul.loadClass("test.ServiceImpl2");
  66.         //返回Class對(duì)象所表示的類的指定公共構(gòu)造方法。    
  67.         Constructor ctr=c.getConstructor(InvocationHandle.class);
  68.         //使用此 Constructor對(duì)象ctr表示的構(gòu)造方法來創(chuàng)建該構(gòu)造方法的聲明類的新實(shí)例,并用指定的初始化參數(shù)初始化該實(shí)例
  69.         Object m = ctr.newInstance(h);
  70.         return m;
  71.     }
  72. }

五、總結(jié)

  1、所謂的動(dòng)態(tài)代理就是這樣一種class,它是在運(yùn)行時(shí)生成的class,在生成它時(shí)你必須提供一組interface給它,然后改 class就宣稱它實(shí)現(xiàn)了這些interface,但是其實(shí)它不會(huì)替你作實(shí)質(zhì)性的工作,而是根據(jù)你在生成實(shí)例時(shí)提供的參數(shù)handler(即 InvocationHandler接口的實(shí)現(xiàn)類),由這個(gè)Handler來接管實(shí)際的工作。

  2、Proxy的設(shè)計(jì)使得它只能支持interface的代理,Java的繼承機(jī)制注定了動(dòng)態(tài)代理類無法實(shí)現(xiàn)對(duì)class的動(dòng)態(tài)代理,因?yàn)槎嗬^承在Java中本質(zhì)上就行不通。


網(wǎng)站名稱:Java動(dòng)態(tài)代理的實(shí)現(xiàn)機(jī)制
標(biāo)題URL:http://www.dlmjj.cn/article/djjsdge.html