日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

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

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
除了JDK、CGLIB,還有3種類代理方式

 五種類代理的方式

不出意外,你可能只知道兩種類代理的方式。一種是JDK自帶的,另外一種是CGLIB。

創(chuàng)新互聯(lián)公司專注為客戶提供全方位的互聯(lián)網(wǎng)綜合服務(wù),包含不限于成都網(wǎng)站制作、成都網(wǎng)站建設(shè)、前郭網(wǎng)絡(luò)推廣、微信小程序定制開發(fā)、前郭網(wǎng)絡(luò)營銷、前郭企業(yè)策劃、前郭品牌公關(guān)、搜索引擎seo、人物專訪、企業(yè)宣傳片、企業(yè)代運營等,從售前售中售后,我們都將竭誠為您服務(wù),您的肯定,是我們最大的嘉獎;創(chuàng)新互聯(lián)公司為所有大學生創(chuàng)業(yè)者提供前郭建站搭建服務(wù),24小時服務(wù)熱線:18982081108,官方網(wǎng)址:www.cdcxhl.com

我們先定義出一個接口和相應的實現(xiàn)類,方便后續(xù)使用代理類在方法中添加輸出信息。

「定義接口」

 
 
 
 
  1. public interface IUserApi { 
  2.  
  3.     String queryUserInfo(); 
  4.  

「實現(xiàn)接口」

 
 
 
 
  1. public class UserApi implements IUserApi { 
  2.  
  3.     public String queryUserInfo() { 
  4.         return "沉淀、分享、成長,讓自己和他人都能有所收獲!"; 
  5.     } 
  6.  

好!接下來我們就給這個類方法使用代理加入一行額外輸出的信息。

0. 先補充一點反射的知識

 
 
 
 
  1. @Test 
  2. public void test_reflect() throws Exception { 
  3.     Class clazz = UserApi.class; 
  4.     Method queryUserInfo = clazz.getMethod("queryUserInfo"); 
  5.     Object invoke = queryUserInfo.invoke(clazz.newInstance()); 
  6.     System.out.println(invoke); 
  • 點評:有代理地方幾乎就會有反射,他們是一套互相配合使用的功能類。在反射中可以調(diào)用方法、獲取屬性、拿到注解等相關(guān)內(nèi)容。這些都可以與接下來的類代理組合使用,完成各種框架中的技術(shù)場景。

1. JDK代理方式

 
 
 
 
  1. public class JDKProxy { 
  2.  
  3.     public static  T getProxy(Class clazz) throws Exception { 
  4.         ClassLoader classLoader = Thread.currentThread().getContextClassLoader(); 
  5.         return (T) Proxy.newProxyInstance(classLoader, new Class[]{clazz}, new InvocationHandler() { 
  6.             public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { 
  7.                 System.out.println(method.getName() + " 你被代理了,By JDKProxy!"); 
  8.                 return "沉淀、分享、成長,讓自己和他人都能有所收獲!"; 
  9.             } 
  10.         }); 
  11.     } 
  12.  
  13.  
  14. @Test 
  15. public void test_JDKProxy() throws Exception { 
  16.     IUserApi userApi = JDKProxy.getProxy(IUserApi.class); 
  17.     String invoke = userApi.queryUserInfo(); 
  18.     logger.info("測試結(jié)果:{}", invoke); 
  19.  
  20. /** 
  21.  * 測試結(jié)果: 
  22.  *  
  23.  * queryUserInfo 你被代理了,By JDKProxy! 
  24.  * 19:55:47.319 [main] INFO  org.itstack.interview.test.ApiTest - 測試結(jié)果: 沉淀、分享、成長,讓自己和他人都能有所收獲! 
  25.  * 
  26.  * Process finished with exit code 0 
  27.  */ 
  • 指數(shù):??
  • 場景:中間件開發(fā)、設(shè)計模式中代理模式和裝飾器模式應用
  • 點評:這種JDK自帶的類代理方式是非常常用的一種,也是非常簡單的一種。基本會在一些中間件代碼里看到例如:數(shù)據(jù)庫路由組件、Redis組件等,同時我們也可以使用這樣的方式應用到設(shè)計模式中。

2. CGLIB代理方式

 
 
 
 
  1. public class CglibProxy implements MethodInterceptor { 
  2.     public Object newInstall(Object object) { 
  3.         return Enhancer.create(object.getClass(), this); 
  4.     } 
  5.     public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { 
  6.         System.out.println("我被CglibProxy代理了"); 
  7.         return methodProxy.invokeSuper(o, objects); 
  8.     } 
  9.  
  10. @Test 
  11. public void test_CglibProxy() throws Exception { 
  12.     CglibProxy cglibProxy = new CglibProxy(); 
  13.     UserApi userApi = (UserApi) cglibProxy.newInstall(new UserApi()); 
  14.     String invoke = userApi.queryUserInfo(); 
  15.     logger.info("測試結(jié)果:{}", invoke); 
  16.  
  17. /** 
  18.  * 測試結(jié)果: 
  19.  *  
  20.  * queryUserInfo 你被代理了,By CglibProxy! 
  21.  * 19:55:47.319 [main] INFO  org.itstack.interview.test.ApiTest - 測試結(jié)果:  沉淀、分享、成長,讓自己和他人都能有所收獲! 
  22.  * 
  23.  * Process finished with exit code 0 
  24.  */ 
  • 場景:Spring、AOP切面、鑒權(quán)服務(wù)、中間件開發(fā)、RPC框架等
  • 點評:CGLIB不同于JDK,它的底層使用ASM字節(jié)碼框架在類中修改指令碼實現(xiàn)代理,所以這種代理方式也就不需要像JDK那樣需要接口才能代理。同時得益于字節(jié)碼框架的使用,所以這種代理方式也會比使用JDK代理的方式快1.5~2.0倍。

3. ASM代理方式

 
 
 
 
  1. public class ASMProxy extends ClassLoader { 
  2.  
  3.     public static  T getProxy(Class clazz) throws Exception { 
  4.  
  5.         ClassReader classReader = new ClassReader(clazz.getName()); 
  6.         ClassWriter classWriter = new ClassWriter(classReader, ClassWriter.COMPUTE_MAXS); 
  7.  
  8.         classReader.accept(new ClassVisitor(ASM5, classWriter) { 
  9.             @Override 
  10.             public MethodVisitor visitMethod(int access, final String name, String descriptor, String signature, String[] exceptions) { 
  11.  
  12.                 // 方法過濾 
  13.                 if (!"queryUserInfo".equals(name)) 
  14.                     return super.visitMethod(access, name, descriptor, signature, exceptions); 
  15.  
  16.                 final MethodVisitor methodVisitor = super.visitMethod(access, name, descriptor, signature, exceptions); 
  17.  
  18.                 return new AdviceAdapter(ASM5, methodVisitor, access, name, descriptor) { 
  19.  
  20.                     @Override 
  21.                     protected void onMethodEnter() { 
  22.                         // 執(zhí)行指令;獲取靜態(tài)屬性 
  23.                         methodVisitor.visitFieldInsn(Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;"); 
  24.                         // 加載常量 load constant 
  25.                         methodVisitor.visitLdcInsn(name + " 你被代理了,By ASM!"); 
  26.                         // 調(diào)用方法 
  27.                         methodVisitor.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", "(Ljava/lang/String;)V", false); 
  28.                         super.onMethodEnter(); 
  29.                     } 
  30.                 }; 
  31.             } 
  32.         }, ClassReader.EXPAND_FRAMES); 
  33.  
  34.         byte[] bytes = classWriter.toByteArray(); 
  35.  
  36.         return (T) new ASMProxy().defineClass(clazz.getName(), bytes, 0, bytes.length).newInstance(); 
  37.     } 
  38.  
  39.  
  40. @Test 
  41. public void test_ASMProxy() throws Exception { 
  42.     IUserApi userApi = ASMProxy.getProxy(UserApi.class); 
  43.     String invoke = userApi.queryUserInfo(); 
  44.     logger.info("測試結(jié)果:{}", invoke); 
  45.  
  46. /** 
  47.  * 測試結(jié)果: 
  48.  *  
  49.  * queryUserInfo 你被代理了,By ASM! 
  50.  * 20:12:26.791 [main] INFO  org.itstack.interview.test.ApiTest - 測試結(jié)果: 沉淀、分享、成長,讓自己和他人都能有所收獲! 
  51.  * 
  52.  * Process finished with exit code 0 
  53.  */ 
  • 場景:全鏈路監(jiān)控、破解工具包、CGLIB、Spring獲取類元數(shù)據(jù)等
  • 點評:這種代理就是使用字節(jié)碼編程的方式進行處理,它的實現(xiàn)方式相對復雜,而且需要了解Java虛擬機規(guī)范相關(guān)的知識。因為你的每一步代理操作,都是在操作字節(jié)碼指令,例如:Opcodes.GETSTATIC、Opcodes.INVOKEVIRTUAL,除了這些還有小200個常用的指令。但這種最接近底層的方式,也是最快的方式。所以在一些使用字節(jié)碼插裝的全鏈路監(jiān)控中,會非常常見。

4. Byte-Buddy代理方式

 
 
 
 
  1. public class ByteBuddyProxy { 
  2.  
  3.     public static  T getProxy(Class clazz) throws Exception { 
  4.  
  5.         DynamicType.Unloaded dynamicType = new ByteBuddy() 
  6.                 .subclass(clazz) 
  7.                 .method(ElementMatchers.named("queryUserInfo")) 
  8.                 .intercept(MethodDelegation.to(InvocationHandler.class)) 
  9.                 .make(); 
  10.  
  11.         return (T) dynamicType.load(Thread.currentThread().getContextClassLoader()).getLoaded().newInstance(); 
  12.     } 
  13.  
  14.  
  15. @RuntimeType 
  16. public static Object intercept(@Origin Method method, @AllArguments Object[] args, @SuperCall Callable callable) throws Exception { 
  17.     System.out.println(method.getName() + " 你被代理了,By Byte-Buddy!"); 
  18.     return callable.call(); 
  19.  
  20. @Test 
  21. public void test_ByteBuddyProxy() throws Exception { 
  22.     IUserApi userApi = ByteBuddyProxy.getProxy(UserApi.class); 
  23.     String invoke = userApi.queryUserInfo(); 
  24.     logger.info("測試結(jié)果:{}", invoke); 
  25.  
  26. /** 
  27.  * 測試結(jié)果: 
  28.  *  
  29.  * queryUserInfo 你被代理了,By Byte-Buddy! 
  30.  * 20:19:44.498 [main] INFO  org.itstack.interview.test.ApiTest - 測試結(jié)果: 沉淀、分享、成長,讓自己和他人都能有所收獲! 
  31.  * 
  32.  * Process finished with exit code 0 
  33.  */ 
  • 場景:AOP切面、類代理、組件、監(jiān)控、日志
  • 點評:Byte Buddy 也是一個字節(jié)碼操作的類庫,但 Byte Buddy 的使用方式更加簡單。無需理解字節(jié)碼指令,即可使用簡單的 API 就能很容易操作字節(jié)碼,控制類和方法。比起JDK動態(tài)代理、cglib,Byte Buddy在性能上具有一定的優(yōu)勢。「另外」,2015年10月,Byte Buddy被 Oracle 授予了 Duke's Choice大獎。該獎項對Byte Buddy的“ Java技術(shù)方面的巨大創(chuàng)新 ”表示贊賞。

5. Javassist代理方式

 
 
 
 
  1. public class JavassistProxy extends ClassLoader { 
  2.  
  3.     public static  T getProxy(Class clazz) throws Exception { 
  4.  
  5.         ClassPool pool = ClassPool.getDefault(); 
  6.         // 獲取類 
  7.         CtClass ctClass = pool.get(clazz.getName()); 
  8.         // 獲取方法 
  9.         CtMethod ctMethod = ctClass.getDeclaredMethod("queryUserInfo"); 
  10.         // 方法前加強 
  11.         ctMethod.insertBefore("{System.out.println(\"" + ctMethod.getName() + " 你被代理了,By Javassist\");}"); 
  12.  
  13.         byte[] bytes = ctClass.toBytecode(); 
  14.  
  15.         return (T) new JavassistProxy().defineClass(clazz.getName(), bytes, 0, bytes.length).newInstance(); 
  16.     } 
  17.  
  18.  
  19. @Test 
  20. public void test_JavassistProxy() throws Exception { 
  21.     IUserApi userApi = JavassistProxy.getProxy(UserApi.class) 
  22.     String invoke = userApi.queryUserInfo(); 
  23.     logger.info("測試結(jié)果:{}", invoke); 
  24.  
  25. /** 
  26.  * 測試結(jié)果: 
  27.  *  
  28.  * queryUserInfo 你被代理了,By Javassist 
  29.  * 20:23:39.139 [main] INFO  org.itstack.interview.test.ApiTest - 測試結(jié)果: 沉淀、分享、成長,讓自己和他人都能有所收獲! 
  30.  * 
  31.  * Process finished with exit code 0 
  32.  */ 
  • 場景:全鏈路監(jiān)控、類代理、AOP
  • 點評:Javassist 是一個使用非常廣的字節(jié)碼插裝框架,幾乎一大部分非入侵的全鏈路監(jiān)控都是會選擇使用這個框架。因為它不想ASM那樣操作字節(jié)碼導致風險,同時它的功能也非常齊全。另外,這個框架即可使用它所提供的方式直接編寫插裝代碼,也可以使用字節(jié)碼指令進行控制生成代碼,所以綜合來看也是一個非常不錯的字節(jié)碼框架。

四、總結(jié)

  • 代理的實際目的就是通過一些技術(shù)手段,替換掉原有的實現(xiàn)類或者給原有的實現(xiàn)類注入新的字節(jié)碼指令。而這些技術(shù)最終都會用到一些框架應用、中間件開發(fā)以及類似非入侵的全鏈路監(jiān)控中。
  • 一個技術(shù)棧深度的學習能讓你透徹的了解到一些基本的根本原理,通過這樣的學習可以解惑掉一些似懂非懂的疑問,也可以通過這樣技術(shù)的拓展讓自己有更好的工作機會和薪資待遇。
  • 這些技術(shù)學起來并不會很容易,甚至可能還有一些燒腦。但每一段值得深入學習的技術(shù)都能幫助你突破一定階段的技術(shù)瓶頸。

當前名稱:除了JDK、CGLIB,還有3種類代理方式
新聞來源:http://www.5511xx.com/article/cdhschp.html