java - 如何方便 Netbeans Designer 加载使用 hashmap 进行枚举反向查找的 JPanel?

标签 java enums hashmap netbeans-8 designer

这已成为 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 );

尚未执行,displayMapnull。此外,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

可能相关的问题:

关于java - 如何方便 Netbeans Designer 加载使用 hashmap 进行枚举反向查找的 JPanel?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44103796/

相关文章:

java - 应用程序崩溃后,Android 蓝牙连接不会关闭

java - 在Java中显式初始化String和Enum变量?

python - 迭代 pandas DataFrame 中的行并创建一个字典

java - ListActivity SimpleAdapter 数据显示不正确

java - 我已经编写了一个 Java servlet 代码来下载文件但出现错误

java - 提高应用速度的技巧

Java 在不应该的情况下对除法结果进行四舍五入

符合单一协议(protocol)的 Swift Enum 关联值

java - 打印枚举变量和值

java.lang.NullPointerException 输出词频-逆文档频率(tfidf)矩阵 java