这已成为 Dorothy Dix因为我在提出这个问题时找到了问题的根源。尽管如此,我还是决定发布它,因为它让我们办公室的每个人都感到困惑,而且我在谷歌或 stackoverflow 搜索中找不到任何有用的信息。这是一个很好的例子,一旦您提出正确的问题,您就可能会看到曙光。
虽然标题可能听起来很复杂,但这是一个非常简单的问题,似乎没有答案。其中心是一个枚举
:
状态
public enum Status
{
INVALID ( "INVALID" ),
ISSUED ( "Issued" ),
CANCELLED ( "Cancelled");
private final String displayName;
private final static Map<String,Status> displayMap = new HashMap( 4 );
private Status( String display ){
this.displayName = display;
mapDisplayName( this.displayName, this );
}
public String getDisplayName(){
return displayName;
}
public static Status parseString( String statusStr ) {
return displayMap.get( statusStr );
}
private static void mapDisplayName( final String displayName, final Status state ){
displayMap.put( displayName, state );
}
}
当然,这个想法是使用displayMap
作为反向查找。与 getDisplayName()
方法完全无关。
此枚举的 getDisplayName()
调用在子面板中用于初始化与组合框一起使用的静态数组,例如:
public class JPanelStatus extends javax.swing.JPanel {
private final String[] STATUS_LABELS = {
Status.ISSUED .getDisplayName(),
Status.CANCELLED .getDisplayName()
};
public JPanelStatus(){
initComponents();
:
jComboBoxStatus.setModel( new DefaultComboBoxModel<>( STATUS_LABELS ) );
:
}
:
}
主 JPanel 中引用了该内容。当我在 Netbeans Designer 中查看此 JPanelStatus
子面板时,它工作正常。 [预览设计
] 功能也是如此。
但是,当我加载主表单时,它失败并且异常显示初始化失败:
java.lang.NoClassDefFoundError: Could not initialize class au.com.project.State
at au.com.project.client.JPanelStatus.<init>(JPanelStatus.java:35)
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:62)
at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
at java.lang.reflect.Constructor.newInstance(Constructor.java:423)
... ... ...
Netbeans IDE 日志添加了以下额外信息:-p
INFO: Could not initialize class au.com.project.State
通过消除过程——注释掉不相关的代码——我发现一旦我注释掉 State 枚举中的 HashMap put()
调用,表单就会加载。
“这很有趣。”,我说。它看起来像是 put()
的副作用。在远处它是——一个小Spock在没有 JPanel 和 Netbeans 的情况下,这很快就在命令行中给了我同样的错误。
该错误是由我尝试在 Enum 构造函数中使用 HashMap 引起的。它不会像写的那样工作。
所以我改变了标题来解决真正的问题——实际上,如何使用 HashMap 来对枚举进行反向查找?
最佳答案
问题来自于如何在 Status
枚举中声明 HashMap,因为枚举是如何初始化的。
Java 枚举中的第一件事必须是值列表,此处为:“INVALID”、“ISSUED”和“CANCELLED”。接下来每个人都需要知道的是,有一个 secret Java 阶段,它在对象创建(类或枚举)期间首先运行。 Init 是愚蠢的,通过声明性代码先来先服务线性运行。
枚举的前 3 个语句调用构造函数 -- 这意味着语句:
private final static Map<String,Status> displayMap = new HashMap( 4 );
尚未执行,displayMap
为null。此外,static { }
block 以相同的 1-2-3-... 顺序执行,也不起作用。
遗憾的是,Netbeans/Designer 堆栈跟踪或 IDE 日志均未报告 NullPointerException——单元测试却报告了 NullPointerException。一旦你有了 NPE,它就会集中你的注意力。当第一次调用构造函数时,displayMap
未初始化。
解决方案:displayMap不能是static final
,因为你不能在构造函数中初始化静态成员。它必须在第一次调用时使用所示示例的某些变体进行初始化:
状态
public enum Status
{
INVALID ( "INVALID" ),
ISSUED ( "Issued" ),
CANCELLED ( "Cancelled");
private static Map<String,Status> displayMap;
private Status( String display ){
this.displayName = display;
mapDisplayName( this.displayName, this );
}
:
private static void mapDisplayName( final String displayName, final Status state ){
if( null == displayMap ){
displayMap = new HashMap( 7 );
}
displayMap.put( displayName, state );
}
}
然后一切都进行得很顺利。
警告:
不要在 displayMap 声明中分配 null
-- 这会适得其反:
if( null == displayMap ){...}
block 在第一次调用 Enum 构造函数期间成功分配了 HashMap。- 处理完所有枚举值声明后。
- Init 将为声明的变量调用任何初始化。
- 如果声明了
displayMap = null;
,它将用一个新的空 HashMap 替换具有 3 个值的填充 HashMap。 grrr
可能相关的问题:
- Reverse lookup for Java Enums with more than one value per key/constant? ...我觉得 HashMap 可以更轻松地实现多个字符串的反向查找。
关于java - 如何方便 Netbeans Designer 加载使用 hashmap 进行枚举反向查找的 JPanel?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44103796/