成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java動態代理的實現機制

開發 后端
代理是一種設計模式,其目的是為其他對象提供一個代理以控制對某個對象的訪問,代理類負責為委托類預處理消息,過濾消息并轉發消息以及進行消息被委托類執行后的后續處理。為了保持行為的一致性,代理類和委托類通常會實現相同的接口。

一、概述

  代理是一種設計模式,其目的是為其他對象提供一個代理以控制對某個對象的訪問,代理類負責為委托類預處理消息,過濾消息并轉發消息以及進行消息被委托類執行后的后續處理。為了保持行為的一致性,代理類和委托類通常會實現相同的接口。

  按照代理的創建時期,代理類可分為兩種:

  • 靜態代理:由程序員創建代理類或特定工具自動生成源代碼再對其編譯,也就是說在程序運行前代理類的.class文件就已經存在。

  • 動態代理:在程序運行時運用反射機制動態創建生成。

  下面在將動態代理的實現機制之前先簡單介紹一下靜態代理。

二、靜態代理

  上面說過,代理類和委托類一般都要實現相同的接口,下面先定義這個接口:

 

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

  委托類作為接口的一種實現,定義如下:

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

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

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

  編寫測試類: 

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

  運行測試程序,結果如下圖:

  從上面的代碼可以看到,靜態代理類只能為特定的接口服務,如果要服務多類型的對象,就要為每一種對象進行代理。我們就會想是否可以通過一個代理類完成全部的代理功能,于是引入的動態代理的概念。

三、動態代理

  Java的動態代理主要涉及兩個類,Proxy和InvocationHandler。

  Proxy:提供了一組靜態方法來為一組接口動態地生成代理類及其對象。 

  1. // 方法 1: 該方法用于獲取指定代理對象所關聯的調用處理器 
  2. static InvocationHandler getInvocationHandler(Object proxy)  
  3.  
  4. // 方法 2:該方法用于獲取關聯于指定類裝載器和一組接口的動態代理類的類對象 
  5. static Class getProxyClass(ClassLoader loader, Class[] interfaces)  
  6.  
  7. // 方法 3:該方法用于判斷指定類對象是否是一個動態代理類 
  8. static boolean isProxyClass(Class cl)  
  9.  
  10. // 方法 4:該方法用于為指定類裝載器、一組接口及調用處理器生成動態代理類實例 
  11. static Object newProxyInstance(ClassLoader loader, Class[] interfaces,InvocationHandler h) 

  InvocationHandler:它是調用處理器接口,自定義了一個invok方法,用于集中處理在動態代理類對象上的方法調用,通常在該方法中實現對委托類的代理訪問

  1. // 該方法負責集中處理動態代理類上的所有方法調用。***個參數既是代理類實例,第二個參數是被調用的方法對象// 第三個方法是調用參數。調用處理器根據這三個參數進行預處理或分派到委托類實例上發射執行 
  2. Object invoke(Object proxy, Method method, Object[] args) 

  實現Java的動態代理,具體有以下四個步驟:

  1. 通過實現InvocationHandler接口創建自己的調用處理器

  2. 通過為Proxy類指定ClassLoader對象和一組interface來創建動態代理類

  3. 通過反射機制獲得動態代理類的構造函數,其***參數類型是調用處理器類接口類型

  4. 通過構造函數創建動態代理類實例,構造時調用處理器對象作為參數被傳入

  下面根據上述的四個步驟來實現自己的動態代理的示例:

  接口和接口的實現類(即委托類)跟上面靜態代理的代碼一樣,這里我們來實現InvocationHandler接口創建自己的調用處理器

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

  編寫測試類: 

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

  運行測試程序,結果同靜態代理。我們可以看到上述代碼并沒有我們之前說的步驟2和3,這是因為Prox的靜態方法newProxyInstance已經為我們封裝了這兩個步驟。具體的內部實現如下:

  1. // 通過 Proxy 為包括 Interface 接口在內的一組接口動態創建代理類的類對象 
  2. Class clazz = Proxy.getProxyClass(classLoader, new Class[] { Interface.class, ... });  
  3.  
  4. // 通過反射從生成的類對象獲得構造函數對象 
  5. Constructor constructor = clazz.getConstructor(new Class[] { InvocationHandler.class });  
  6.  
  7. // 通過構造函數對象創建動態代理類實例 
  8. Interface Proxy = (Interface)constructor.newInstance(new Object[] { handler }); 

  newProxyInstance函數的內部實現為:

  1. public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h)throws IllegalArgumentException 
  2.   { 
  3.              //檢查h不為空,否則拋異常 
  4.             Objects.requireNonNull(h); 
  5.             //獲得與制定類裝載器和一組接口相關的代理類類型對象 
  6.             final Class<?>[] intfs = interfaces.clone(); 
  7.              
  8.             //檢查接口類對象是否對類裝載器可見并且與類裝載器所能識別的接口類對象是完全相同的 
  9.             final SecurityManager sm = System.getSecurityManager();     
  10.             if (sm != null)  
  11.             { 
  12.                 checkProxyAccess(Reflection.getCallerClass(), loader, intfs); 
  13.             } 
  14.             //獲得與制定類裝載器和一組接口相關的代理類類型對象 
  15.             Class<?> cl = getProxyClass0(loader, intfs); 
  16.             try 
  17.             { 
  18.                 if (sm != null)  
  19.                 { 
  20.                     checkNewProxyPermission(Reflection.getCallerClass(), cl); 
  21.                 } 
  22.                 // 通過反射獲取構造函數對象并生成代理類實例 
  23.                 final Constructor<?> cons = cl.getConstructor(constructorParams); 
  24.                 final InvocationHandler ih = h; 
  25.                 if (!Modifier.isPublic(cl.getModifiers()))  
  26.                 { 
  27.                     AccessController.doPrivileged(new PrivilegedAction<Void>()  
  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.  } 

四、模擬實現Proxy類

  根據上面的原理介紹,我們可以自己模擬實現Proxy類:

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

五、總結

  1、所謂的動態代理就是這樣一種class,它是在運行時生成的class,在生成它時你必須提供一組interface給它,然后改 class就宣稱它實現了這些interface,但是其實它不會替你作實質性的工作,而是根據你在生成實例時提供的參數handler(即 InvocationHandler接口的實現類),由這個Handler來接管實際的工作。

  2、Proxy的設計使得它只能支持interface的代理,Java的繼承機制注定了動態代理類無法實現對class的動態代理,因為多繼承在Java中本質上就行不通。

 

責任編輯:王雪燕 來源: 博客園
相關推薦

2015-09-24 08:55:14

Java動態代理擴展

2015-09-24 08:54:36

java動態代理

2010-04-21 09:26:54

Java動態代理

2012-02-08 10:12:19

Java反射

2017-05-11 21:30:01

Android動態代理ServiceHook

2011-04-06 11:41:25

Java動態代理

2012-01-09 11:26:15

Java

2012-02-08 10:37:42

Java反射

2015-09-22 11:09:47

Java 8動態代理

2023-04-06 00:11:12

Java接口開發

2012-08-28 10:59:26

JavaJava動態代理Proxy

2021-07-06 06:39:22

Java靜態代理動態代理

2009-06-22 15:10:00

java 編程AOP

2022-11-15 09:57:51

Java接口

2016-12-14 14:29:30

Java動態綁定機制

2017-10-12 14:56:11

2011-03-23 10:40:51

java代理模式

2023-12-06 08:23:44

代理模式設計模式

2022-04-02 07:52:47

DubboRPC調用動態代理

2023-02-24 07:42:30

Java動態代理
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日批的视频 | 亚洲免费一区二区 | 亚洲高清一区二区三区 | av男人天堂影院 | 精区3d动漫一品二品精区 | 亚洲国产精品99久久久久久久久 | 日韩一区在线观看视频 | 中文字幕在线视频免费视频 | 日本精品视频在线 | 亚洲三区在线观看 | 6080yy精品一区二区三区 | 亚洲 欧美 日韩在线 | 午夜性色a√在线视频观看9 | 国产一区 | 亚洲播放 | 国产在线视频一区二区董小宛性色 | 99国产精品99久久久久久粉嫩 | 国产一区二区在线免费 | 中文字幕精品一区久久久久 | 亚洲a视 | 精品国产不卡一区二区三区 | 日韩成人一区 | 91久久夜色 | 精品国产一区二区国模嫣然 | 中文字幕精品一区二区三区精品 | 尤物视频在线免费观看 | 久久免费视频网 | 久久国产精彩视频 | 成人福利片 | 在线视频成人 | 91精品国产91久久久久久吃药 | 欧美激情一区二区三区 | 欧一区二区 | 亚洲成人免费观看 | 久久高清| 亚洲欧洲精品成人久久奇米网 | 久久专区| 欧洲尺码日本国产精品 | 日韩一区中文字幕 | 在线播放国产一区二区三区 | 天天综合网永久 |