我有一个使用自定义系统类加载器的应用程序,由Djava.system.class.loader=class
参数设置。应用程序使用 RMI。eclipse 一切正常,但是如果我导出一个可运行的 jar,就会发生这个错误:
Caused by: java.lang.SecurityException: SHA MessageDigest not available
at sun.rmi.server.Util.computeMethodHash(Unknown Source)
at sun.rmi.server.UnicastServerRef$HashToMethod_Maps.computeValue(Unknown Source)
at sun.rmi.server.UnicastServerRef$HashToMethod_Maps.computeValue(Unknown Source)
at sun.rmi.server.WeakClassHashMap.get(Unknown Source)
at sun.rmi.server.UnicastServerRef.exportObject(Unknown Source)
at sun.rmi.registry.RegistryImpl.setup(Unknown Source)
at sun.rmi.registry.RegistryImpl.<init>(Unknown Source)
at java.rmi.registry.LocateRegistry.createRegistry(Unknown Source)
我很自然地试图解决这个问题,并MessageDigest md = MessageDigest.getInstance("SHA");
在我的 main.js 中写了一个简单的第一行。基本上同样的情况发生。确切的错误是:
java.security.NoSuchAlgorithmException: SHA MessageDigest not available
at sun.security.jca.GetInstance.getInstance(Unknown Source)
at java.security.Security.getImpl(Unknown Source)
at java.security.MessageDigest.getInstance(Unknown Source)
然后我尝试使用一个简单的 ClassLoader 作为 SystemClassloader。现在看起来像这样:
public class RootClassLoader extends ClassLoader{
@Override
public Class<?> loadClass(String name) throws ClassNotFoundException {
return super.loadClass(name);
}
public RootClassLoader(ClassLoader loader){
super(loader);
}
}
仍然会发生同样的情况。如果我将这个场景重建为一个新项目,一切都会按预期工作,所以很明显还有其他东西坏了,但现在我不知道是什么。
我不在程序中的任何地方使用安全管理器,而是使用 oracle 的标准 jre7。
有什么我可能忽略的吗?
更新:
即使我loadClass
根本不覆盖该方法,也会发生错误......它似乎只是因为有一个非标准的系统类加载器而失败,不管它实际上做了什么
更新 2:
我列出了所有可用的安全提供程序/算法。当我在 Eclipse 中执行代码时(仍然使用自定义类加载器),一切似乎都井井有条,但是在执行可运行的 jar 时,SUN version 1.7
似乎缺少完整的提供程序。没有自定义类加载器,一切都是一样的。在仅包含上述场景(并且有效)的环境中,一切看起来都一样,并且每个星座都有SUN version 1.7
提供者。
更新 3:
显示-verbose:class
包sun.security.provider
和更多内容没有被加载。我不知道为什么,因为这两种选择都是从jsse.jar
丢失包的来源加载类,并且都打印[Opened C:\Program Files\Java\jre7\lib\jsse.jar]
(更新4:实际上它确实被加载了。只是很久以后。甚至sun.security.provider.SHA
在两个变体中都加载了。这更奇怪)