当前位置: 首页 > news >正文

java17学习笔记-Deprecate the Applet API for Removal

默认情况下封装 JDK 的大多数内部 API,以便它们 在编译时无法访问,并为将来的版本做准备,其中 它们在运行时将无法访问。确保关键的、广泛使用的 内部 API 不会封装,因此它们可以访问,直到 其全部或大部分功能都存在受支持的替换项。

一些流行的库使用非标准、不稳定和不受支持的API,这些API是JDK的内部实现细节,从未打算用于外部使用。在模块化JDK(JEP 200)中,通过利用模块系统(JEP 261)限制对这些API的访问可以提高平台的完整性和安全性,因为许多内部API定义了特权、安全敏感的操作。从长远来看,这一变化将降低JDK本身的维护者以及有意或无意使用这些内部API的库和应用程序的维护者所承担的成本。

起因由于模块化的实行

 JEP 200: The Modular JDK

JEP 201: Modular Source Code

JEP 220: Modular Run-Time Images

JEP 260: Encapsulate Most Internal APIs

JEP 261: Module System

JEP 275: Modular Java Application Packaging

JEP 282: jlink: The Java Linker

发展历史

JEP 260: Encapsulate Most Internal APIs

JEP 396: Strongly Encapsulate JDK Internals by Default

JEP 403: Strongly Encapsulate JDK Internals

JEP 260: Encapsulate Most Internal APIs

JDK 9 中未封装的关键内部 API

  • sun.misc.{Signal,SignalHandler}

  • sun.misc.Unsafe (The functionality of many of the methods in this class is available via variable handles (JEP 193).)

  • sun.reflect.Reflection::getCallerClass(int) (The functionality of this method is available in the stack-walking API defined by JEP 259.)

  • sun.reflect.ReflectionFactory

  • com.sun.nio.file.{ExtendedCopyOption,ExtendedOpenOption, ExtendedWatchEventModifier,SensitivityWatchEventModifier}

JEP 396: Strongly Encapsulate JDK Internals by Default

该提案的主要风险是现有的 Java 代码将无法运行。失败的代码类型包括但不限于:

  • 使用 的方法的框架,以便在现有 类加载器。此类框架应改用 ,它已 从 JDK 9 开始可用。protecteddefineClassjava.lang.ClassLoaderjava.lang.invoke.MethodHandles.Lookup::defineClass

  • 使用类来作时区信息的代码。此类代码应改用 API (自 JDK 8 以来可用)。sun.util.calendar.ZoneInfojava.time

  • 使用包处理 SQL 行集的代码。此类代码应改用包从 JDK 7 开始可用。com.sun.rowsetjavax.sql.rowset

  • 使用包处理源代码的工具 法典。 此类工具应改用 、 和 API,这些 API 自JDK 6 以来可用。com.sun.tools.javac.*javax.toolsjavax.lang.modelcom.sun.source.*

  • 使用类的代码 以生成自签名证书。目前还没有标准 此功能的 API(尽管请求已被 已提交);同时,开发人员可以使用 包含此功能的现有第三方库。sun.security.tools.keytool.CertAndKeyGen

  • 使用 JDK 的 Xerces XML 处理器内部副本的代码。 此类代码应改用 Xerces 库的独立副本,可从 Maven Central 获得。
  • 使用 JDK 内部版本的 ASM 字节码的代码 图书馆。此类代码应改用 ASM 的独立副本 库,可从 Maven Central 获得。

此更改的影响示例

  • 使用直接 访问 JDK 的内部 API 将不再默认工作。 例如

    System.out.println(sun.security.util.SecurityConstants.ALL_PERMISSION);
    //Exception in thread "main" java.lang.IllegalAccessError: class Main (in unnamed module @0x4eec7777) cannot access class sun.security.util.SecurityConstants (in module java.base) because module java.base does not export sun.security.util to unnamed module @0x4eec7777
    //	at Main.main(Main.java:23)
  • 默认情况下,使用反射访问导出java.*API的私有字段的代码将不再工作。

    var ks = java.security.KeyStore.getInstance("jceks");
    var f = ks.getClass().getDeclaredField("keyStoreSpi");
    f.setAccessible(true);
    //Exception in thread "main" java.lang.reflect.InaccessibleObjectException: Unable to make field private final java.security.KeyStoreSpi java.security.KeyStore.keyStoreSpi accessible: module java.base does not "opens java.security" to unnamed module @531d72ca
    //	at java.base/java.lang.reflect.AccessibleObject.throwInaccessibleObjectException(AccessibleObject.java:353)
    //	at java.base/java.lang.reflect.AccessibleObject.checkCanSetAccessible(AccessibleObject.java:329)
    //	at java.base/java.lang.reflect.AccessibleObject.checkCanSetAccessible(AccessibleObject.java:277)
    //	at java.base/java.lang.reflect.Field.checkCanSetAccessible(Field.java:179)
    //	at java.base/java.lang.reflect.Field.setAccessible(Field.java:173)
    //	at Main.main(Main.java:22)
  • 默认情况下,使用反射调用导出java.*API的受保护方法的代码将不再工作。

    Method dc = ClassLoader.class.getDeclaredMethod("defineClass",String.class,byte[].class,int.class,int.class);
    dc.setAccessible(true);
    //Exception in thread "main" java.lang.reflect.InaccessibleObjectException: Unable to make protected final java.lang.Class java.lang.ClassLoader.defineClass(java.lang.String,byte[],int,int) throws java.lang.ClassFormatError accessible: module java.base does not "opens java.lang" to unnamed module @f6f4d33
    //	at java.base/java.lang.reflect.AccessibleObject.throwInaccessibleObjectException(AccessibleObject.java:353)
    //	at java.base/java.lang.reflect.AccessibleObject.checkCanSetAccessible(AccessibleObject.java:329)
    //	at java.base/java.lang.reflect.AccessibleObject.checkCanSetAccessible(AccessibleObject.java:277)
    //	at java.base/java.lang.reflect.Method.checkCanSetAccessible(Method.java:182)
    //	at java.base/java.lang.reflect.Method.setAccessible(Method.java:176)
    //	at Main.main(Main.java:35)

JEP 403: Strongly Encapsulate JDK Internals

本JEP的目标、非目标、动机、风险和假设部分与JEP 396基本相同,但为方便读者,此处转载。

http://www.dtcms.com/a/338755.html

相关文章:

  • C语言基础:(十八)C语言内存函数
  • 连接远程服务器上的 jupyter notebook,解放本地电脑
  • 计算机毕设推荐:痴呆症预测可视化系统Hadoop+Spark+Vue技术栈详解
  • 生成式AI的能力边界与职业重构:从“百科实习生“到人机协作增强器
  • 人工智能学派简介
  • 当宠物机器人装上「第六感」:Deepoc 具身智能如何重构宠物机器人照看逻辑
  • Python字符串变量插值深度解析:从基础到高级工程实践
  • 安装DDNS-go
  • 【部署相关】DockerKuberbetes常用命令大全(速查+解释)
  • 便携式科研土壤监测仪:让土壤检测走进 “轻时代”
  • 大数据MapReduce架构:分布式计算的经典范式
  • 【MySQL】--- 库表操作
  • Python + 淘宝 API 开发:自动化采集商品数据的完整流程​
  • Redis(11)如何通过命令行操作Redis?
  • 对象创建过程
  • 《算法导论》第 32 章 - 字符串匹配
  • 大数据云原生是什么
  • 中国技术引领人工心脏变革——欧洲心脏与心力衰竭大会特别报道
  • 思科语音系统简要了解
  • 【科研绘图系列】R语言绘制多种小提琴和云雨图
  • 期权小故事:王安石变法与期权
  • electron进程间通信- 渲染进程与主进程双向通信
  • GitHub 热榜项目 - 日榜(2025-08-19)
  • 从现场到云端的“通用语”:Kepware 在工业互联中的角色、使用方法与本土厂商(以胡工科技为例)的差异与优势
  • AiPPT怎么样?好用吗?
  • Ubuntu22系统上源码部署LLamaFactory+微调模型 教程【亲测成功】
  • Linux下编译ARPACK
  • 【Git Submodules 与微前端架构技术指南】
  • git仓库和分支的关系
  • 主从切换是怎么保证数据一致的?从库为什么会延迟