我的数百名用户中只有一位用户无法启动我的Java桌面应用。只有三分之一的时间才开始。另外三分之二的时间在启动时抛出NullPointerException:
Exception in thread "AWT-EventQueue-0" java.lang.NullPointerException
at java.util.Hashtable.put(Hashtable.java:394)
at javax.swing.JEditorPane.registerEditorKitForContentType(JEditorPane.java:1327)
at javax.swing.JEditorPane.registerEditorKitForContentType(JEditorPane.java:1309)
at javax.swing.JEditorPane.loadDefaultKitsIfNecessary(JEditorPane.java:1387)
at javax.swing.JEditorPane.getKitTypeRegistry(JEditorPane.java:1344)
at javax.swing.JEditorPane.getEditorKitClassNameForContentType(JEditorPane.java:1340)
at javax.swing.JTextPane.<init>(JTextPane.java:76)
at myapp.Launcher$1.run(Launcher.java:13)
at java.awt.event.InvocationEvent.dispatch(InvocationEvent.java:209)
at java.awt.EventQueue.dispatchEvent(EventQueue.java:633)
at java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThread.java:296)
at java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.java:211)
at java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThread.java:201)
at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:196)
at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:188)
at java.awt.EventDispatchThread.run(EventDispatchThread.java:122)
我已经跟踪堆栈跟踪,发现原因是
Thread.currentThread().getContextClassLoader()
JEditorPane中的返回null。
谷歌搜索显示,这是一个零星的,非常罕见的,神秘的问题,影响了一些人。我的问题是,我可以做什么作为解决方法?如果我在创建EditorPane之前调用它,这可能会有效:
Thread.currentThread().setContextClassLoader(MyClass.class.getClassLoader());
但是我并不像我想的那样真正理解类加载器(并且我试图更好地理解它们)。我觉得改变EDT中的contextClassLoader会产生不良后果。
任何想法我能做什么?
编辑:我与熟悉Java ClassLoaders的人有一些通信。这似乎是一个模糊的ClassLoader竞争条件。也就是说,Java中的一个错误。答案 0 :(得分:4)
Thread.currentThread().getContextClassLoader()
如果JEditorPane.registerEditorKitForContentType
中的代码未检查上述代码中的空返回值,则这是JEditorPane
中的错误。请注意MyClass.class.getClassLoader()
may also return null。您唯一可以依赖的是system ClassLoader。
为调用设置上下文ClassLoader
的模式通常如下所示:
Thread thread = Thread.currentThread();
ClassLoader old = thread.getContextClassLoader();
thread.setContextClassLoader(fooClassLoader);
try {
// do call that depends on context ClassLoader
} finally {
thread.setContextClassLoader(old);
}
应通过setContextClassLoader
设置的值取决于使用它的代码的意图以及您正在运行的ClassLoader
框架的设计。
在一个独立的应用程序中,你可能只需使用这个ClassLoader
(将ref传递给当前类)就可以了:
private ClassLoader findClassLoaderForContext(Class<?> c) {
ClassLoader context = Thread.currentThread().getContextClassLoader();
ClassLoader me = c.getClassLoader();
ClassLoader system = ClassLoader.getSystemClassLoader();
return (context == null) ? (me == null) ? system : me : context;
}
在ClassLoader敏感的插件框架中(Java EE服务器将是一个主要的例子),理解加载方案的性质和用法是值得的。