bog*_*gus 20 java patch swingx
类BasicLabelUI中的javax /秋千/ PLAF /基本是通过影响已确认的bug.在我的应用程序中,我需要固定版本提供的功能(为v9提交).由于法律和技术原因,我仍然受限于受影响的JDK版本.
我的方法是在我的项目中创建一个包javax/swing/plaf/basic,包含固定版本.
如何强制我的项目支持我在已安装的JDK中的缺陷类上包含的类的版本?
这必须具有一定的便携性,因为固定类也必须在客户端工作,并且必须忽略JDK安装中的缺陷类.因此,我不想修改JDK,而是绕过这个特定的类.
Raf*_*ter 19
正如其他的答案中提到,你可以在课程的理论解压缩到JVM的rt.jar的文件,并用兼容bugfixed版本替换该文件.
任何类的Java类库(例如Swing的类)都由引导类加载器加载,该类加载器从此rt.jar中查找其类.你一般不能在前面加上类此类路径,而不将其添加到该文件.有一个(非标准)VM选项
-Xbootclasspath/jarWithPatchedClass.jar:path
Run Code Online (Sandbox Code Playgroud)
您可以在其中添加包含修补版本的jar文件,但这不一定适用于任何Java虚拟机.此外,部署更改此行为的应用程序是非法的!正如官方文件中所述:
不要部署使用此选项的应用程序来覆盖rt.jar中的类,因为这违反了Java Runtime Environment二进制代码许可证.
但是,如果您将一个类附加到引导类加载器(通过使用检测API而不使用非标准API可能的话),运行时仍将加载原始类,因为引导类加载器在这种情况下首先搜索rt.jar.因此,如果不修改此文件,就不可能"破坏"破碎的类.
最后,使用修补文件分发VM,即将其放入客户的生产系统中始终是非法的.许可协议明确规定您需要
[...]分发[Java运行时]完整且未经修改,仅作为applet和应用程序的一部分捆绑
因此,不建议更改您分发的VM,因为在发现此情况时可能会遇到法律后果.
当然,理论上您可以构建自己的OpenJDK版本,但是当您分发它时,您无法再调用二进制Java,并且我认为您的客户不会通过您在答案中建议的内容来实现这一点.根据经验,许多安全环境在执行之前计算二进制的哈希值,这将禁止两种调整执行VM的方法.
最简单的解决方案可能是创建您在启动时添加到VM进程的Java代理.最后,这与将库添加为类路径依赖项非常相似:
java -javaagent:bugFixAgent.jar -jar myApp.jar
Run Code Online (Sandbox Code Playgroud)
Java代理能够在应用程序启动时替换类的二进制表示,因此可以更改错误方法的实现.
在您的情况下,代理看起来类似于以下内容,您需要将修补的类文件包含为ressource:
public static class BugFixAgent {
public static void premain(String args, Instrumentation inst) {
inst.addClassFileTransformer(new ClassFileTransformer() {
@Override
public byte[] transform(ClassLoader loader,
String className,
Class<?> classBeingRedefined,
ProtectionDomain protectionDomain,
byte[] classfileBuffer) {
if (className.equals("javax/swing/plaf/basic/BasicLabelUI")) {
return patchedClassFile; // as found in the repository
// Consider removing the transformer for future class loading
} else {
return null; // skips instrumentation for other classes
}
}
});
}
}
Run Code Online (Sandbox Code Playgroud)
javadoc java.lang.instrumentation包提供了有关如何构建和实现Java代理的详细说明.使用此方法,您可以在不违反许可协议的情况下使用相关类的固定版本.
根据经验,Java代理是修复第三方库和Java类库中的临时错误的好方法,无需在代码中部署更改,甚至不需要为客户部署新版本.事实上,这是使用Java代理的典型用例.