首页 > Java > java教程 > 正文

Java 17 中使用反射修改 final 字段

DDD
发布: 2025-08-15 20:10:29
原创
876人浏览过

java 17 中使用反射修改 final 字段

本文将介绍如何在 Java 17 中使用反射来修改 final 字段的值。在 Java 12 及更高版本中,直接通过反射修改 Field 对象的 modifiers 字段的方式已经失效。本文提供了一种适用于 Java 17 的解决方案,该方案利用 VarHandle 和 JVM 启动参数来克服 Java 的模块化限制。

使用 VarHandle 修改 final 字段

Java 17 引入了更强的模块化和封装性,直接修改 Field 对象的 modifiers 字段不再可行。一种替代方案是使用 VarHandle。VarHandle 提供了一种更安全、更灵活的方式来访问和操作类的字段,包括私有字段。

以下代码展示了如何使用 VarHandle 修改 final 字段:

import java.lang.invoke.MethodHandles;
import java.lang.invoke.VarHandle;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;

class Foo {
    private final String bar;

    public Foo(String bar) {
        this.bar = bar;
    }

    public String getBar() {
        return this.bar;
    }
}

public class Example {

    public static void main(String[] args) {
        Foo foo = new Foo("foobar");
        System.out.println(foo.getBar());

        try {
            Field field = foo.getClass().getDeclaredField("bar");
            field.setAccessible(true);

            VarHandle MODIFIERS;
            var lookup = MethodHandles.privateLookupIn(Field.class, MethodHandles.lookup());
            MODIFIERS = lookup.findVarHandle(Field.class, "modifiers", int.class);
            MODIFIERS.set(field, field.getModifiers() & ~Modifier.FINAL);

            field.set(foo, "new value"); // 修改 final 字段的值

        } catch (Exception e) {
            e.printStackTrace();
        }

        System.out.println(foo.getBar());
    }
}
登录后复制

代码解释:

立即学习Java免费学习笔记(深入)”;

AI Sofiya
AI Sofiya

一款AI驱动的多功能工具

AI Sofiya 109
查看详情 AI Sofiya
  1. 获取 Field 对象: 首先,通过 getDeclaredField() 方法获取要修改的 final 字段的 Field 对象。
  2. 设置 accessible: 调用 setAccessible(true) 允许访问私有字段。
  3. 获取 VarHandle: 使用 MethodHandles.privateLookupIn() 获取 Field 类的私有查找上下文。然后,使用 findVarHandle() 方法查找名为 "modifiers" 的 VarHandle。
  4. 移除 FINAL 修饰符: 使用 MODIFIERS.set() 方法,通过位运算 field.getModifiers() & ~Modifier.FINAL 移除 final 修饰符。
  5. 修改字段值: 移除 final 修饰符后,就可以使用 field.set() 方法修改字段的值。

JVM 启动参数

为了使上述代码正常工作,需要在启动 JVM 时添加以下参数:

--add-opens=java.base/java.lang.reflect=ALL-UNNAMED
--add-opens=java.base/java.net=ALL-UNNAMED
登录后复制

这些参数允许代码访问 java.base 模块中的 java.lang.reflect 和 java.net 包,这是使用 VarHandle 所必需的。 如果不添加这些参数,将会抛出java.lang.IllegalAccessException异常。

如何添加 JVM 启动参数:

  • IDE (例如 IntelliJ IDEA, Eclipse): 在 Run/Debug Configurations 中,找到 VM options 选项,添加上述参数。
  • 命令行: 在运行 Java 程序时,直接在 java 命令后添加参数,例如:java --add-opens=java.base/java.lang.reflect=ALL-UNNAMED Example.java

注意事项

  • 安全风险: 使用反射修改 final 字段可能会破坏类的封装性,导致不可预测的行为。请谨慎使用此技术。
  • 模块化: Java 的模块化系统对反射访问进行了限制。需要使用 --add-opens 参数来允许访问特定的模块和包。
  • 适用范围: 此方法适用于非静态 final 字段。修改静态 final 字段通常是不可能的,因为它们的值在类加载时就已经确定。

总结

虽然在 Java 17 中修改 final 字段变得更加困难,但通过使用 VarHandle 和适当的 JVM 启动参数,仍然可以实现这一目标。然而,请务必谨慎使用此技术,并充分了解其潜在风险。建议在确实需要绕过 final 限制的情况下才使用此方法,并仔细测试代码以确保其正确性和稳定性。

以上就是Java 17 中使用反射修改 final 字段的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号