【深度思考】聊聊JDK动态代理原理

1. 示例

首先,定义一个接口:

public interface Staff {
    void work();
}

然后,新增一个类并实现上面的接口:

public class Coder implements Staff {
    @Override
    public void work() {
        System.out.println("认真写bug……");
    }
}

假设现在有这么一个需求:在不改动以上类代码的前提下,对该方法增加一些前置操作或者后置操作。

接下来就来讲解下,如何使用JDK动态代理来实现这个需求。

首先,自定义一个调用处理器,实现java.lang.reflect.InvocationHandler接口并重写invoke方法:

public class AttendanceInvocationHandler implements InvocationHandler {
    private final Object target;

    public AttendanceInvocationHandler(Object target) {
        this.target = target;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("上班打卡……");

        Object invoke = method.invoke(target, args);

        System.out.println("下班打卡……");

        return invoke;
    }
}

重点看下Object invoke = method.invoke(target, args);,该行代码会执行真正的目标方法,在这前后,我们可以添加一些增强逻辑。

然后,新建个测试类,看下JDK动态代理如何使用:

public class JdkProxyTest {
    public static void main(String[] args) {
        Coder coder = new Coder();
        AttendanceInvocationHandler h = new AttendanceInvocationHandler(coder);
        // 创建代理对象
        Object proxyInstance = Proxy.newProxyInstance(coder.getClass().getClassLoader(),
                coder.getClass().getInterfaces(),
                h);
        Staff staff = (Staff) proxyInstance;
        staff.work();
    }
}

运行以上代码,效果如下图所示:

【深度思考】聊聊JDK动态代理原理

从运行结果可以看出,在目标方法的前后,执行了自定义的操作。

2. 原理

这里理解2个概念,目标对象和代理对象,

目标对象是真正要调用的对象,上面示例中的Coder类就是目标对象,

代理对象是JDK自动生成的对象,在代理对象内部会去调用目标对象的目标方法。

JDK动态代理的核心就是上面示例中的Proxy.newProxyInstance方法,方法签名如下图所示:

【深度思考】聊聊JDK动态代理原理

第1个参数传入的是目标对象的ClassLoader,第2个参数传入的是目标对象的接口信息,第3个参数传入的是自定义的InvocationHandler。

然后看下该方法的实现逻辑,先看第1处重点:

【深度思考】聊聊JDK动态代理原理

注释翻译过来是:查找或者生成指定的代理类。

该方法会生成代理类的字节码文件(也可能是从缓存中读取),核心逻辑在ProxyClassFactory类的apply方法中,

该方法中定义了生成的代理类的包名以及文件名:

【深度思考】聊聊JDK动态代理原理

【深度思考】聊聊JDK动态代理原理

因此默认情况下,自动生成的代理类名称是com.sun.proxy.$Proxy0

该方法最后会生成代理类的字节码,默认情况下不会保存到文件系统,但可以通过参数指定保存到文件系统:

【深度思考】聊聊JDK动态代理原理

【深度思考】聊聊JDK动态代理原理

可以看出,保存不保存到文件系统,受saveGeneratedFiles的影响,其定义如下所示:

private static final boolean saveGeneratedFiles = (Boolean)AccessController.doPrivileged(new GetBooleanAction("sun.misc.ProxyGenerator.saveGeneratedFiles"));

所以可以通过指定sun.misc.ProxyGenerator.saveGeneratedFiles参数来让生成的代理类字节码文件保存到文件系统中。

然后看第2处重点:

【深度思考】聊聊JDK动态代理原理

先是获取构造函数,然后是生成代理类对象的实例。

3. 为什么必须要基于接口?

思考一个问题,为什么JDK动态代理必须要基于接口,带着这个问题,我们看下动态生成的代理类com.sun.proxy.$Proxy0长什么样子?

JVM参数里添加参数-Dsun.misc.ProxyGenerator.saveGeneratedFiles=true,然后启动上面示例中的测试代码:

【深度思考】聊聊JDK动态代理原理

【深度思考】聊聊JDK动态代理原理

生成的代理类字节码文件保存在项目根目录下的com/sun/proxy目录下:

【深度思考】聊聊JDK动态代理原理

在IDEA中打开后,如下图所示:

【深度思考】聊聊JDK动态代理原理

在静态代码块中,对静态变量m0、m1、m2、m3进行了赋值,其中m3是要执行的目标方法。

在构造方法中,执行的是super(var1);,也就是父类Proxy的构造方法:

【深度思考】聊聊JDK动态代理原理

该方法是将我们自定义的InvocationHandler赋值给了父类的变量h。

而以下测试代码实际执行的是代理类$Proxy0里的work方法:

Staff staff = (Staff) proxyInstance;
staff.work();

【深度思考】聊聊JDK动态代理原理

代理类$Proxy0里的work方法实际执行的是自定义InvocationHandler里的invoke方法:

【深度思考】聊聊JDK动态代理原理

因此在执行目标方法前后,执行了自定义的前置操作和后置操作。

了解了这个调用过程,就理解了为什么JDK动态代理必须要基于接口,因为动态生成的代理类已经继承了类java.lang.reflect.Proxy

Java又是单继承的,如果想要继续对类进行扩展,只能通过实现接口的方式。

文章持续更新,欢迎关注微信公众号「申城异乡人」第一时间阅读!

原文链接:https://www.cnblogs.com/zwwhnly/p/17324797.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:【深度思考】聊聊JDK动态代理原理 - Python技术站

(0)
上一篇 2023年4月17日
下一篇 2023年4月17日

相关文章

  • 利用SpringDataJPA开启审计功能,自动保存操作人操作时间

    利用SpringDataJPA开启审计功能可以方便地对每次数据操作进行记录,包括操作人、操作时间等信息。这样做有利于数据追溯和安全性控制。下面是实现该功能的步骤: 添加必要的依赖 在项目的pom.xml文件中添加以下依赖: <dependency> <groupId>org.springframework.boot</group…

    Java 2023年5月20日
    00
  • java编程下字符串的16位,32位md5加密实现方法

    Java编程下字符串的16位、32位MD5加密实现方法 MD5(Message-Digest Algorithm 5)是一种哈希算法,常用于确保文件完整性以及验证数据传输完整性。MD5加密后,得到的结果是一个128位(32个十六进制字符)的字符串,可以选择后16位或者后32位作为加密结果。在Java编程中,我们可以使用Java自带的MessageDigest…

    Java 2023年5月27日
    00
  • Java中遍历数组使用foreach循环还是for循环?

    在Java中遍历数组可以使用foreach循环和for循环,那么这两种方式有什么异同呢?如何选择使用哪种方式呢?下面就来详细讲解。 foreach循环 foreach循环也叫增强for循环,可以在数组或集合中遍历元素。这种循环方式相比传统的for循环有下面几个优点: 简洁明了,代码可读性更好。 不需要手动维护计数器,只需要直接遍历即可。 可以避免数组下标越界…

    Java 2023年5月26日
    00
  • C++字符串的处理详解

    C++字符串的处理详解 在C++中,字符串是一种很重要的数据类型。可以使用以下两种方法来处理字符串: 1. 使用C风格的字符串处理方式 C风格的字符串其实是一个字符数组,字符串的结束标志是’\0’。 字符串的定义: char str[10]; //定义一个长度为10的字符数组作为字符串 字符串的输出: printf("%s", str);…

    Java 2023年5月27日
    00
  • 什么是同步?

    以下是关于同步的完整使用攻略: 什么是同步? 同步是指多个线程之间按照一定的顺序执行,以避免出现数据竞争和不一致的情况。在多线程编程中,同步是非常重要的,因为多个线程同时访问共享资源时,可能会导致数据的不一致性和程序的错误。 同步的实现方式 同步可以通过以下几种方式来实现: synchronized关键字:synchronized关键字可以用来修饰方法或代码…

    Java 2023年5月12日
    00
  • 基于JAVA代码 获取手机基本信息(本机号码,SDK版本,系统版本,手机型号)

    要获取手机的基本信息,可以使用Android的系统API。下面是获取本机号码、SDK版本、系统版本和手机型号的完整攻略: 准备工作 首先,我们需要为项目添加依赖项,具体依赖项如下: dependencies { implementation ‘com.android.support:support-v4:28.0.0’ } 以上例子使用的是support库的…

    Java 2023年5月24日
    00
  • SpringBoot 整合Security权限控制的初步配置

    下面是 “SpringBoot 整合Security权限控制的初步配置”的完整攻略,包含了基础概念、示例程序与注意事项。 1. 简介 Spring Security 是一个安全框架,提供了认证、授权、攻击防护等一系列的安全功能,是目前比较流行的开源 Java 安全框架之一。 Spring Security 采用基于过滤器的方式实现安全控制,对 URL 进行拦…

    Java 2023年6月3日
    00
  • Java基于递归解决全排列问题算法示例

    Java基于递归解决全排列问题的算法是比较经典的算法问题,通过递归实现,可以快速地求解全排列问题,下面将详细介绍基于递归解决全排列问题的算法示例。 什么是全排列 全排列就是将一组数按照一定顺序排列,即所有的数字都被使用了,仅次序不同,就认为是一种不同的排列方式。例如,对于数字1,2,3的全排列,可以得到如下6种排列方式: 1 2 3 1 3 2 2 1 3 …

    Java 2023年5月19日
    00
合作推广
合作推广
分享本页
返回顶部