在同一包中对 sun.* 类进行子类化会导致 IllegalAccessError

问题描述 投票:0回答:1

前言:

  1. 我要向你展示的是错误的,我很清楚我做了这种愚蠢的事情来破坏封装是多么糟糕。
  2. 我不想解决任何更一般的 I/O 问题。这只是一个实验。

我正在尝试对

sun.nio.ch.SourceChannelImpl
进行子类化,它是 JDK 中存在的包私有构造函数(在 rt.jar 中),所以我必须在
sun.nio.ch
包中创建它。

这是我的子类:

package sun.nio.ch;
import java.io.FileDescriptor;
import java.nio.channels.spi.SelectorProvider;
class MySourceChannel extends SourceChannelImpl {
  public MySourceChannel(SelectorProvider sp, FileDescriptor fd) {
    super(sp, fd);
  }
}

这是我的简单测试:

package sun.nio.ch;
import java.io.FileDescriptor;
public class Main {
  public static void main(String[] args) {
    new MySourceChannel(null, FileDescriptor.in);
  }
}

这是失败的:

Exception in thread "main" java.lang.IllegalAccessError: class sun.nio.ch.MySourceChannel cannot access its superclass sun.nio.ch.SourceChannelImpl
        at java.lang.ClassLoader.defineClass1(Native Method)
        at java.lang.ClassLoader.defineClass(ClassLoader.java:800)
        at java.security.SecureClassLoader.defineClass(SecureClassLoader.java:142)
        at java.net.URLClassLoader.defineClass(URLClassLoader.java:449)
        at java.net.URLClassLoader.access$100(URLClassLoader.java:71)
        at java.net.URLClassLoader$1.run(URLClassLoader.java:361)
        at java.net.URLClassLoader$1.run(URLClassLoader.java:355)
        at java.security.AccessController.doPrivileged(Native Method)
        at java.net.URLClassLoader.findClass(URLClassLoader.java:354)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:425)
        at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:358)
        at sun.nio.ch.Main.main(Main.java:5)

这可能不是 you can't Define class in JDK package XYZ ((java|sun).*) 类型的问题,否则我会得到

java.lang.SecurityException: Prohibited package name: XYZ

Main
类在这个包中运行良好。

我还尝试通过设置

Policy
允许一切来禁用安全检查,但这也没有帮助。我也尝试过
System.setSecurityManager(null);
(我不确定这是否真的会禁用它),但它也没有帮助。

有什么问题吗?请问我该如何解决?

我已经使用 JDK 1.7.0_45、Oracle 和 OpenJDK 进行了尝试。

java inheritance visibility sun
1个回答
6
投票

SourceChannelImpl
是一个“包私有”类。在 JVM 中,包始终由单个类加载器加载。如果你有两个同名的包被不同的类加载器加载,那么它们就不是同一个包。

您可以通过使用

-Xbootclasspath/a:mybootspath
在引导类加载器中加载部分或全部代码来解决此问题。

© www.soinside.com 2019 - 2024. All rights reserved.