Java反射机制剖析:深度剖析动态代理原理及总结

开发 后端
通过这篇文章对动态代理进行了深度剖析,现在想起来还感觉非常有意思,这里面其实最根本的机制就是反射机制,运行时动态实例化任何一个类,并且调用它的具体细节。

动态代理类原理 示例代码参见《Java反射机制剖析:简单谈谈动态代理》)

a) 理解上面的动态代理示例流程

 

b) 代理接口实现类源代码剖析

咱们一起来剖析一下代理实现类($Proxy0)的源代码和整个动态代理的流程。

$Proxy0生成的代码如下:

  1. import java.lang.reflect.InvocationHandler;  
  2. import java.lang.reflect.Method;  
  3. import java.lang.reflect.Proxy;  
  4. import java.lang.reflect.UndeclaredThrowableException;  
  5.    
  6. public final class $Proxy0 extends Proxy implements Manager {  
  7.    
  8. private static Method m1;  
  9. private static Method m0;  
  10. private static Method m3;  
  11. private static Method m2;  
  12.    
  13. static {  
  14.    try {  
  15.     m1 = Class.forName("java.lang.Object").getMethod("equals",  
  16.       new Class[] { Class.forName("java.lang.Object") });  
  17.     m0 = Class.forName("java.lang.Object").getMethod("hashCode",  
  18.       new Class[0]);  
  19.     m3 = Class.forName("com.ml.test.Manager").getMethod("test",  
  20.       new Class[0]);  
  21.     m2 = Class.forName("java.lang.Object").getMethod("toString",  
  22.       new Class[0]);  
  23.    } catch (NoSuchMethodException nosuchmethodexception) {  
  24.     throw new NoSuchMethodError(nosuchmethodexception.getMessage());  
  25.    } catch (ClassNotFoundException classnotfoundexception) {  
  26.     throw new NoClassDefFoundError(classnotfoundexception.getMessage());  
  27.    }  
  28. }  
  29.    
  30. public $Proxy0(InvocationHandler invocationhandler) {  
  31.    super(invocationhandler);  
  32. }  
  33.    
  34. @Override 
  35. public final boolean equals(Object obj) {  
  36.    try {  
  37.     return ((Boolean) super.h.invoke(this, m1, new Object[] { obj }))  
  38.       .booleanValue();  
  39.    } catch (Throwable throwable) {  
  40.     throw new UndeclaredThrowableException(throwable);  
  41.    }  
  42. }  
  43.    
  44. @Override 
  45. public final int hashCode() {  
  46.    try {  
  47.     return ((Integer) super.h.invoke(this, m0, null)).intValue();  
  48.    } catch (Throwable throwable) {  
  49.     throw new UndeclaredThrowableException(throwable);  
  50.    }  
  51. }  
  52.    
  53. public final void test() {  
  54.    try {  
  55.     super.h.invoke(this, m3, null);  
  56.     return;  
  57.    } catch (Error e) {  
  58.    } catch (Throwable throwable) {  
  59.     throw new UndeclaredThrowableException(throwable);  
  60.    }  
  61. }  
  62.    
  63. @Override 
  64. public final String toString() {  
  65.    try {  
  66.     return (String) super.h.invoke(this, m2, null);  
  67.    } catch (Throwable throwable) {  
  68.     throw new UndeclaredThrowableException(throwable);  
  69.    }  
  70. }  

引入眼帘的是这个代理接口实现类实现了业务类的接口(也就是例子中的UserManager接口),又继承了基类Proxy类;

接着就是构造函数,在构造方法中把BusinessHandler传过去,接着$Proxy0调用父类Proxy的构造器,为h赋值(这里要看Proxy的构造方法);

随后看到的就是这个类重写了Proxy类的Equals、hashCode、toString方法,又实现了业务类接口的方法(即UserManager的test方法),具体重写和实现都是用到的super.h.invoke(即Proxy.h.invoke)这个方法。

 

简单分析完这个代理接口实现类,咱们下面来整体看一下这个动态代理是怎么实现的:

首先客户端初始化了BusinessHandler类,调用这个类的newProxyInstance(new UserManagerImpl())方法来初始化了上面的代理接口实现类;

接下来代理接口实现类通过构造函数把BusinessHandler传过去(也就是代码中的this),并通过Proxy的构造函数给h赋值;

随后再客户端就能实例化出代理接口实现类$Proxy0,我们把它强制转换为业务实现接口(UserManager)类型的(为什么要强制转换,这里非常有意思,如果不强制转换就会报错,这里很好解释,因为当前的环境根本不会知道这个代理接口实现类$Proxy0既继承Proxy又实现业务实现接口UserManager,但是强制转换成UserManager它是可以做到的,因为当前环境中就有UserManager。这就是反射的厉害之处,可以在运行时动态调用任何一个类并可以使用这个类的具体细节。);

之后当我们调用test方法的时候其实是调用了$Proxy0中的test方法,这个方法的实现是通过Proxy.h的invoke方法实现的(即调用了BusinessHandler.invoke方法);

之后在调用了Method的invoke方法(这时的参数是this,和args)。

这样就调用了UserManagerImpl的对应方法,之后返回给客户端。

到此就完成了整个的调用关系。

 

反射,反射,程序员的快乐

通过上篇文章对动态代理进行了深度剖析,现在想起来还感觉非常有意思,这里面其实最根本的机制就是反射机制,运行时动态实例化任何一个类,并且调用它的具体细节。现在反看动态代理的示例,其实发现这里最关键的还是在就在Proxy.newProxyInstance(..)方法执行时生成了$Proxy0的内存字节码这一点上,当我们有了内存字节码,我们的反射就会大显威力,这样才有了我们之后的一系列的调用关系。

通过反射机制的分析和动态代理示例的剖析,发现编程是一件多么有意思的事情,以至于我们沉浸其中不能自拔。

***总结一下:反射,反射,程序员的快乐!

原文链接:http://blog.csdn.net/lfsf802/article/details/7239799

【编辑推荐】

  1. Java反射机制剖析:简单谈谈动态代理
  2. Java反射机制剖析:功能以及举例
  3. Java反射机制剖析:定义和API
  4. Java远程方法调用RMI
  5. Java处理字符串搜索嵌套结构的方法
责任编辑:林师授 来源: lfsf802的博客
相关推荐

2012-02-08 10:12:19

Java反射

2012-02-08 09:44:52

Java反射

2012-02-08 09:53:25

Java反射

2010-09-17 12:39:51

JAVA反射机制

2024-03-12 12:57:07

Redis主从架构

2021-01-10 10:30:24

站点隔离Chrome漏洞

2015-09-24 08:55:14

Java动态代理扩展

2015-09-24 08:54:36

java动态代理

2019-11-27 14:41:50

Java技术语言

2021-01-13 09:42:58

站点隔离Chrome漏洞

2010-04-21 09:26:54

Java动态代理

2010-07-12 21:44:51

HART协议

2010-01-13 11:14:06

C++虚表

2022-09-27 18:56:28

ArrayList数组源代码

2024-02-05 19:06:04

DartVMGC流程

2016-09-02 12:34:52

2019-11-25 16:05:20

MybatisPageHelpeJava

2012-08-17 10:55:21

IBMdW

2024-04-12 14:04:17

机器学习DNN

2016-12-19 14:35:32

Spark Strea原理剖析数据
点赞
收藏

51CTO技术栈公众号