JAVA 动态代理学习记录
打算用JAVA实现一个简单的RPC框架,看完RPC参考代码之后,感觉RPC的实现主要用到了两个方面的JAVA知识:网络通信和动态代理。因此,先补补动态代理的知识。---多看看代码中写的注释
在动态代理中,首先定义一个接口,这个接口中声明的方法 是 真实类需要实现的,真实类实现该方法来提供具体的操作。
public interface Subject {
public abstract void request();
}
public class RealSubject extends Subject{//具体实现类
public RealSubject(){ }
public void request(){
System.out.println("From Real Subject");
}
}
有了具体实现类,现在就需要代理类了,具体实现类在本例中为RealSubject.java ,它就是 被代理的类,即代理类 代理的“家伙”就是 具体实现类。
代理类需要实现 InvocationHandler 接口,为什么呢?先了解下JAVA动态代理中需要用到的一个接口:InvocationHandler 和 一个类: java.lang.reflect.Proxy
InvocationHandler接口中只有一个invoke方法,该方法需要三个参数:
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{ proxy:需要代理的真实类的对象
method:“待调用的真实对象的某个方法”
args:调用该方法时需要的参数
InvocationHandler的作用就是将需要代理的类的对象作为构造函数的参数传给它,即传给代理类,这样代理类就知道自己代理的对象是什么了。DynamicSubject implements InvocationHandler
//给构造方法传递不同的类型的 被代理的对象,就可以 实现 动态代理--即在运行时确定被代理的对象的类型
public DynamicSubject(Object obj){
sub = obj;
}
同时,在客户端代码中进行方法调用时,会自动执行InvocationHandler接口的invoke方法,从而由InvocationHandler接口的invoke方法 中的 method.invoke() 再去执行真正的被代理类的方法。
method.invoke(sub, args);//调用 实际 的方法,调用被代理的对象的方法,即RealSubject.request()
看看JDK中java.lang.reflect.Method 中的invoke() 方法的定义:对带有指定参数的指定对象调用由此 Method
对象表示的底层方法。----即调用 指定对象 sub ,指定参数 args 所代表的方法。
再看看Proxy类的作用,Proxy类主要用来在Clinet代码中生成一个动态的对象。该对象调用方法来开始进行动态代理。
/* 第一个参数指定哪个 ClassLoader对象来加载我们的代理对象
* 第二个参数 为代理对象提供的接口是真实对象所实现的接口,表示我要代理的是该真实对象,这样我就能调用这组接口中的方法了
* 第三个参数handler, 这里将这个代理对象关联到了上方的 InvocationHandler 这个对象上,这样,当执行下条实际方法调用语句时,就可以
* 知道委托的是哪个InvocationHandler 了,进程就会自动执行该 InvocationHandler 的 invoke方法
*/ Subject subject = (Subject)Proxy.newProxyInstance(cls.getClassLoader(), cls.getInterfaces(), handler); /*
* 当该语句执行时,它会委托到InvocationHandler 类中的 invoke方法,并执行 method.invoke()进行实际调用
* 注意:InvocationHandler.invoke()中有两条输出语句,运行Client后在控制台中看到了其输出结果,说明该方法被委托执行了
*/
subject.request();
完整的动态代理类的实现代码如下:
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method; public class DynamicSubject implements InvocationHandler{
private Object sub;//被 代理的对象,它是一个Object类型的对象,说明,被 代理的对象是可以动态改变的 public DynamicSubject(){ } //给构造方法传递不同的类型的 被代理的对象,就可以 实现 动态代理--即在运行时确定被代理的对象的类型
public DynamicSubject(Object obj){
sub = obj;
} public Object invoke(Object proxy, Method method, Object[] args) throws Throwable{
System.out.println("before calling " + method); /*
* 在构造方法中获得了被代理的对象RealSubject sub
*/
method.invoke(sub, args);//调用 实际 的方法,调用被代理的对象的方法,即RealSubject.request() System.out.println("after calling " + method);
return null;
}
}
完整的客户端实现代码如下:Client.java中的 subject.request(); 表示开始执行代理调用。可以从代码中看出,subject 对象是由 Proxy 动态生成的。
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy; public class Client {
public static void main(String[] args) throws Exception{
RealSubject rs = new RealSubject();// 需要 被 代理的类,即客户端现在需要的代理一个类型为 RealSubject 的对象 /*
* 注意,代理类的构造方法的参数为Object类型,说明它可以代理 任意类型的对象---即程序运行前DynamicSubject并不知道需要代理的对象
* AnOtherRealSubject anotherRs = new AnOtherRealSubject();
* InvocationHandler handler = new DynamicSubject(anotherRs);
*
*/
InvocationHandler handler = new DynamicSubject(rs);// 代理类,将需要 被 代理的类 作为 代理类的构造 函数的参数传入 Class cls = rs.getClass(); /*
* Class c = Proxy.getProxyClass(cls.getClassLoader(), cls.getInterfaces());
Constructor ct = c.getConstructor(new Class[]{InvocationHandler.class});
Subject subject = (Subject) ct.newInstance(new Object[]{handler});
*/ /*
* 第二个参数 cls.getInterfaces()... cls 是代表RealSubject类型的 Class对象,参考JDK中Class类的getInterfaces()
*表明:newProxyInstance 知道动态生成的代理对象subject 需要实现哪些接口---需要实现的接口由getInterfaces()指定。
*而cls 是一个代表RealSubject类型的Class对象,RealSubject 实现了 Subject 接口,因此动态生成的subject 对象当然可以
*强制类型转换了。
*/
Subject subject = (Subject)Proxy.newProxyInstance(cls.getClassLoader(), cls.getInterfaces(), handler); /*
* 当该语句执行时,它会委托到InvocationHandler 类中的 invoke方法,并执行 method.invoke()进行实际调用
* 注意:InvocationHandler.invoke()中有两条输出语句,运行Client后在控制台中看到了其输出结果,说明该方法被委托执行了
*/
subject.request(); /*
* 输出:com.sun.proxy.$Proxy0
* 这表明,subject 对象类型是 $Proxy0,而不是 Subject 或 InvocationHandler 类型
* 但是在 22行语句中,却可以将之进行强制类型转换,转成Subject类型
* 原因是:Proxy.newProxyInstance生成的是一个动态对象,即在JVM运行时生成的。在newProxyInstance()的第二个参数上,给它提供了一组接口
* 该代理对象就会实现这组接口,因此也就可以将该对象强制转化为这组接口中的任意一个
*/
System.out.println(subject.getClass().getName());
}
}
整个程序代码参考github: https://github.com/hapjin/JAVA/tree/master/dynamicProxy
JAVA 动态代理学习记录的更多相关文章
- Java动态代理学习【Spring AOP基础之一】
Spring AOP使用的其中一个底层技术就是Java的动态代理技术.Java的动态代理技术主要围绕两个类进行的 java.lang.reflect.InvocationHandler java.la ...
- Java动态代理学习
动态代理类 Java动态代理类位于java.lang.reflect包下,一般主要涉及到以下两个类: 1.Interface InvocationHandler 该接口中仅定义了一个方法: Objec ...
- java jdk动态代理学习记录
转载自: https://www.jianshu.com/p/3616c70cb37b JDK自带的动态代理主要是指,实现了InvocationHandler接口的类,会继承一个invoke方法,通过 ...
- java 动态代理学习(Proxy,InvocationHandler)
前几天看到java的动态代理机制,不知道是啥玩意,然后看了看.死活不知道 invoke(Object proxy, Method m, Object[] args)种的proxy是个什么东西,放在这里 ...
- CgLib动态代理学习【Spring AOP基础之一】
如果不了解JDK中proxy动态代理机制的可以先查看上篇文章的内容:Java动态代理学习[Spring AOP基础之一] 由于Java动态代理Proxy.newProxyInstance()的时候会发 ...
- java 动态代理深度学习(Proxy,InvocationHandler),含$Proxy0源码
java 动态代理深度学习, 一.相关类及其方法: java.lang.reflect.Proxy,Proxy 提供用于创建动态代理类和实例的静态方法.newProxyInstance()返回一个指定 ...
- Java动态代理-->Spring AOP
引述要学习Spring框架的技术内幕,必须事先掌握一些基本的Java知识,正所谓“登高必自卑,涉远必自迩”.以下几项Java知识和Spring框架息息相关,不可不学(我将通过一个系列分别介绍这些Jav ...
- 一文读懂Java动态代理
作者 :潘潘 日期 :2020-11-22 事实上,对于很多Java编程人员来说,可能只需要达到从入门到上手的编程水准,就能很好的完成大部分研发工作.除非自己强主动获取,或者工作倒逼你学习,否则我们好 ...
- Java 动态代理机制详解
在学习Spring的时候,我们知道Spring主要有两大思想,一个是IoC,另一个就是AOP,对于IoC,依赖注入就不用多说了,而对于Spring的核心AOP来说,我们不但要知道怎么通过AOP来满足的 ...
随机推荐
- [转帖] 打开加密SQLite文件的方法
Copy From http://blog.csdn.net/sean4m/article/details/50211565 mark 下 正好工作用到了这个东西. 版本:SQLiteExpertPr ...
- node.js依赖express解析post请求四种数据格式()
分别是这四种: www-form-urlencoded, form-data, application/json, text/xml www-form-urlencoded 这是http的post请求 ...
- DTD举例二
DTD举例二: <!--hbase语句库约束文件--> <!DOCTYPE hbaseGroup [ <!ELEMENT hbaseGroup (statements*)> ...
- python之count()函数
# count()统计字符串中特定单词或短语出现次数(n = 3) strs = 'Good! Today is good day! Good job!' n = strs.lower().count ...
- char类型的数值转换
在视频教程中,你已经认识到了数字类型之间.字符串和其他类型之间的转换.而某些时候,我们还需要将char类型转换为int类型,或者把int类型转换为char类型. 这篇文章,将介绍在代码中虽然不太常用, ...
- BZOJ3419[POI2013]taxis——贪心
题目大意: 一条线段有三个点,0为初始位置,d为出租车总部位置,m为家的位置,人要叫车,有n辆车可以提供,每辆车有一个路程上限,并且都从车站出发,叫的车行驶之后不必须回到车站,问最少叫几辆车. 一定能 ...
- hdu 5126 stars (四维偏序,离线,CDQ套CDQ套树状数组)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5126 思路:支持离线,那么我们可以用两次CDQ分治使四维降为二维,降成二维后排个序用树状数组维护下就好 ...
- day9-13 linux基础
有道云笔记链接 http://note.youdao.com/noteshare?id=207be3d6bd79e9ff2e30b160bca1fd87
- JVM 内存区域 (运行时数据区域)
JVM 内存区域 (运行时数据区域) 链接:https://www.jianshu.com/p/ec479baf4d06 运行时数据区域 Java 虚拟机在执行 Java 程序的过程中会把它所管理的内 ...
- MT【209】打破对称
设正数$a,b,c$满足$ab+bc+ca=47$,求$(a^2+5)(b^2+5)(c^2+5)$的最小值_____ 解:$(a^2+5)(b^2+5)(c^2+5)=(a^2+5)(5(b+c)^ ...