除了JDK、CGLIB,还有3种类代理方式

开发 前端
不出意外,你可能只知道两种类代理的方式。一种是JDK自带的,另外一种是CGLIB。

 五种类代理的方式

不出意外,你可能只知道两种类代理的方式。一种是JDK自带的,另外一种是CGLIB。

我们先定义出一个接口和相应的实现类,方便后续使用代理类在方法中添加输出信息。

「定义接口」

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

「实现接口」

  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<UserApi> clazz = UserApi.class
  4.     Method queryUserInfo = clazz.getMethod("queryUserInfo"); 
  5.     Object invoke = queryUserInfo.invoke(clazz.newInstance()); 
  6.     System.out.println(invoke); 
  • 点评:有代理地方几乎就会有反射,他们是一套互相配合使用的功能类。在反射中可以调用方法、获取属性、拿到注解等相关内容。这些都可以与接下来的类代理组合使用,完成各种框架中的技术场景。

1. JDK代理方式

  1. public class JDKProxy { 
  2.  
  3.     public static <T> 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("测试结果:{}", invoke); 
  19.  
  20. /** 
  21.  * 测试结果: 
  22.  *  
  23.  * queryUserInfo 你被代理了,By JDKProxy! 
  24.  * 19:55:47.319 [main] INFO  org.itstack.interview.test.ApiTest - 测试结果: 沉淀、分享、成长,让自己和他人都能有所收获! 
  25.  * 
  26.  * Process finished with exit code 0 
  27.  */ 

 

  • 指数:⭐⭐
  • 场景:中间件开发、设计模式中代理模式和装饰器模式应用
  • 点评:这种JDK自带的类代理方式是非常常用的一种,也是非常简单的一种。基本会在一些中间件代码里看到例如:数据库路由组件、Redis组件等,同时我们也可以使用这样的方式应用到设计模式中。

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("测试结果:{}", invoke); 
  16.  
  17. /** 
  18.  * 测试结果: 
  19.  *  
  20.  * queryUserInfo 你被代理了,By CglibProxy! 
  21.  * 19:55:47.319 [main] INFO  org.itstack.interview.test.ApiTest - 测试结果:  沉淀、分享、成长,让自己和他人都能有所收获! 
  22.  * 
  23.  * Process finished with exit code 0 
  24.  */ 

 

  • 场景:Spring、AOP切面、鉴权服务、中间件开发、RPC框架等
  • 点评:CGLIB不同于JDK,它的底层使用ASM字节码框架在类中修改指令码实现代理,所以这种代理方式也就不需要像JDK那样需要接口才能代理。同时得益于字节码框架的使用,所以这种代理方式也会比使用JDK代理的方式快1.5~2.0倍。

3. ASM代理方式

  1. public class ASMProxy extends ClassLoader { 
  2.  
  3.     public static <T> 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.                         // 执行指令;获取静态属性 
  23.                         methodVisitor.visitFieldInsn(Opcodes.GETSTATIC, "java/lang/System""out""Ljava/io/PrintStream;"); 
  24.                         // 加载常量 load constant 
  25.                         methodVisitor.visitLdcInsn(name + " 你被代理了,By ASM!"); 
  26.                         // 调用方法 
  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("测试结果:{}", invoke); 
  45.  
  46. /** 
  47.  * 测试结果: 
  48.  *  
  49.  * queryUserInfo 你被代理了,By ASM! 
  50.  * 20:12:26.791 [main] INFO  org.itstack.interview.test.ApiTest - 测试结果: 沉淀、分享、成长,让自己和他人都能有所收获! 
  51.  * 
  52.  * Process finished with exit code 0 
  53.  */ 
  • 场景:全链路监控、破解工具包、CGLIB、Spring获取类元数据等
  • 点评:这种代理就是使用字节码编程的方式进行处理,它的实现方式相对复杂,而且需要了解Java虚拟机规范相关的知识。因为你的每一步代理操作,都是在操作字节码指令,例如:Opcodes.GETSTATIC、Opcodes.INVOKEVIRTUAL,除了这些还有小200个常用的指令。但这种最接近底层的方式,也是最快的方式。所以在一些使用字节码插装的全链路监控中,会非常常见。

4. Byte-Buddy代理方式

  1. public class ByteBuddyProxy { 
  2.  
  3.     public static <T> T getProxy(Class clazz) throws Exception { 
  4.  
  5.         DynamicType.Unloaded<?> dynamicType = new ByteBuddy() 
  6.                 .subclass(clazz) 
  7.                 .method(ElementMatchers.<MethodDescription>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("测试结果:{}", invoke); 
  25.  
  26. /** 
  27.  * 测试结果: 
  28.  *  
  29.  * queryUserInfo 你被代理了,By Byte-Buddy! 
  30.  * 20:19:44.498 [main] INFO  org.itstack.interview.test.ApiTest - 测试结果: 沉淀、分享、成长,让自己和他人都能有所收获! 
  31.  * 
  32.  * Process finished with exit code 0 
  33.  */ 
  • 场景:AOP切面、类代理、组件、监控、日志
  • 点评:Byte Buddy 也是一个字节码操作的类库,但 Byte Buddy 的使用方式更加简单。无需理解字节码指令,即可使用简单的 API 就能很容易操作字节码,控制类和方法。比起JDK动态代理、cglib,Byte Buddy在性能上具有一定的优势。「另外」,2015年10月,Byte Buddy被 Oracle 授予了 Duke's Choice大奖。该奖项对Byte Buddy的“ Java技术方面的巨大创新 ”表示赞赏。

5. Javassist代理方式

  1. public class JavassistProxy extends ClassLoader { 
  2.  
  3.     public static <T> 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("测试结果:{}", invoke); 
  24.  
  25. /** 
  26.  * 测试结果: 
  27.  *  
  28.  * queryUserInfo 你被代理了,By Javassist 
  29.  * 20:23:39.139 [main] INFO  org.itstack.interview.test.ApiTest - 测试结果: 沉淀、分享、成长,让自己和他人都能有所收获! 
  30.  * 
  31.  * Process finished with exit code 0 
  32.  */ 

 

  • 场景:全链路监控、类代理、AOP
  • 点评:Javassist 是一个使用非常广的字节码插装框架,几乎一大部分非入侵的全链路监控都是会选择使用这个框架。因为它不想ASM那样操作字节码导致风险,同时它的功能也非常齐全。另外,这个框架即可使用它所提供的方式直接编写插装代码,也可以使用字节码指令进行控制生成代码,所以综合来看也是一个非常不错的字节码框架。

四、总结

除了JDK、CGLIB,还有3种类代理方式?面试又卡住
  • 代理的实际目的就是通过一些技术手段,替换掉原有的实现类或者给原有的实现类注入新的字节码指令。而这些技术最终都会用到一些框架应用、中间件开发以及类似非入侵的全链路监控中。
  • 一个技术栈深度的学习能让你透彻的了解到一些基本的根本原理,通过这样的学习可以解惑掉一些似懂非懂的疑问,也可以通过这样技术的拓展让自己有更好的工作机会和薪资待遇。
  • 这些技术学起来并不会很容易,甚至可能还有一些烧脑。但每一段值得深入学习的技术都能帮助你突破一定阶段的技术瓶颈。

 

责任编辑:张燕妮 来源: 今日头条
相关推荐

2023-12-06 08:23:44

代理模式设计模式

2024-01-04 07:42:44

JavaCGLIBJDK

2022-09-01 10:40:29

SpringAOPJDK

2015-08-10 14:54:57

公有云云安全数据加密

2022-12-02 08:48:16

CSS置灰网站

2024-07-31 11:26:05

反射BeanXML

2021-07-14 11:07:56

AOPJDKCglib

2009-06-15 14:15:07

Java设计模式Java

2020-06-17 07:56:19

前端存储数据

2016-10-26 08:57:13

HadoopScrapy大数据

2017-01-15 14:18:35

大数据HadoopScrapy

2021-08-30 08:40:02

网页404HTTP

2024-05-23 13:54:40

2011-05-19 15:57:15

测试专家

2019-01-15 18:20:57

2021-09-28 09:52:08

Prometheus开源工具Kubernetes

2012-06-19 09:21:17

BYOD

2015-06-24 15:35:54

2018-10-09 13:23:22

2018-01-24 05:08:24

点赞
收藏

51CTO技术栈公众号