JAVA 动态代理学习记录

简介:

打算用JAVA实现一个简单的RPC框架,看完RPC参考代码之后,感觉RPC的实现主要用到了两个方面的JAVA知识:网络通信和动态代理。因此,先补补动态代理的知识。---多看看代码中写的注释

参考:Java 代理模式与动态代理类  

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


本文转自hapjin博客园博客,原文链接:http://www.cnblogs.com/hapjin/,如需转载请自行联系原作者

相关文章
|
27天前
|
存储 Oracle Java
java零基础学习者入门课程
本课程为Java零基础入门教程,涵盖环境搭建、变量、运算符、条件循环、数组及面向对象基础,每讲配示例代码与实践建议,助你循序渐进掌握核心知识,轻松迈入Java编程世界。
225 0
|
1月前
|
负载均衡 Java API
grpc-java 架构学习指南
本指南系统解析 grpc-java 架构,涵盖分层设计、核心流程与源码结构,结合实战路径与调试技巧,助你从入门到精通,掌握高性能 RPC 开发精髓。
189 7
|
1月前
|
IDE Java 编译器
java编程最基础学习
Java入门需掌握:环境搭建、基础语法、面向对象、数组集合与异常处理。通过实践编写简单程序,逐步深入学习,打牢编程基础。
170 0
|
2月前
|
Java
Java基础学习day08-作业
本作业涵盖Java中Lambda表达式的应用,包括Runnable与Comparator接口的简化实现、自定义函数式接口NumberProcessor进行加减乘及最大值操作,以及通过IntProcessor处理整数数组,实现遍历、平方和奇偶判断等功能,强化函数式编程实践。
68 5
|
2月前
|
Java API 容器
Java基础学习day08-2
本节讲解Java方法引用与常用API,包括静态、实例、特定类型方法及构造器引用的格式与使用场景,并结合代码示例深入解析。同时介绍String和ArrayList的核心方法及其实际应用。
147 1
|
2月前
|
Java 程序员
Java基础学习day08
本节讲解Java中的代码块(静态与实例)及其作用,深入介绍内部类(成员、静态、局部及匿名)的定义与使用,并引入函数式编程思想,重点阐述Lambda表达式及其在简化匿名内部类中的应用。
129 5
|
2月前
|
Java
Java基础学习day07-作业
本作业包含六个Java编程案例:1)动物类继承与多态;2)加油卡支付系统;3)员工管理类设计;4)学生信息统计接口;5)USB设备控制;6)家电智能控制。综合运用抽象类、接口、继承、多态等面向对象技术,强化Java基础编程能力。
171 3
|
2月前
|
Java
Java基础学习day06-作业
本内容为Java基础学习作业,涵盖两个案例:一是通过Card类及其子类GoldenCard、SilverCard实现加油卡系统,体现封装与继承;二是通过Shape类及子类Circle、Rectangle演示多态与方法重写,强化面向对象编程理解。
75 1
|
2月前
|
设计模式 存储 Java
Java基础学习day07
本节讲解Java中的final关键字、单例设计模式、枚举类、抽象类与接口。涵盖常量定义、单例写法(饿汉式/懒汉式)、枚举特点及应用场景,以及抽象类与接口的使用与区别,助力掌握核心面向对象编程思想。
133 1
|
2月前
|
Java
Java基础学习day05-作业
本文为Java基础学习第五天作业,通过五个案例练习类与对象的定义、构造方法、set/get方法及成员方法的应用。涵盖女友、学生、教师、手机和电影等类的设计与测试,强化面向对象编程基础。
80 2