Java Enum静态最终实例变量

时间:2011-06-22 05:22:42

标签: java static enums compiler-errors final

好哇!

此代码工作了一段时间,然后我决定添加默认颜色,它停止工作。我收到以下错误:

1 error found:
File: Status.java  [line: 20]
Error: Status.java:20: illegal reference to static field from initializer

在编译时使用以下代码。

import java.awt.Color;

enum Status
{
  OFF ("Off"),
  TRAINING ("Training", new Color(255, 191, 128)),
  BEGINNER ("Beginner", new Color(128, 255, 138)),
  INTERMEDIATE ("Intermediate", new Color(128, 212, 255)),
  ADVANCED ("Advanced", new Color(255, 128, 128));

  public final String name;
  public final Color color;

  public static final Color defaultColor = Color.WHITE;

  Status(String name)
  {
    this(name, defaultColor);
  }
  Status(String name, Color color)
  {
    this.name = name;
    this.color = color;
  }
}

据我所知,这应该可行,但无论出于什么原因,Java决定抛出错误。 有什么想法吗?

4 个答案:

答案 0 :(得分:27)

defaultColor只会在调用构造函数后初始化 - 因此在此之前它将具有默认值(null)。一种选择是将默认颜色放在嵌套类型中:

import java.awt.Color;

enum Status
{
  OFF ("Off"),
  TRAINING ("Training", new Color(255, 191, 128)),
  BEGINNER ("Beginner", new Color(128, 255, 138)),
  INTERMEDIATE ("Intermediate", new Color(128, 212, 255)),
  ADVANCED ("Advanced", new Color(255, 128, 128));

  public final String name;
  public final Color color;

  Status(String name)
  {
    this(name, Defaults.COLOR);
  }
  Status(String name, Color color)
  {
    this.name = name;
    this.color = color;
  }

  private static class Defaults
  {
     private static Color COLOR = Color.WHITE;
  }
}

当然,如果您只是在代码中引用默认颜色一次,那么您也可以在构造函数调用中对其进行硬编码:

Status(String name)
{
  this(name, Color.WHITE);
}

答案 1 :(得分:12)

必须首先初始化枚举常量。要初始化它们,必须调用构造函数。第一个构造函数引用一个静态字段,该字段在调用时可能无法初始化。

答案 2 :(得分:6)

Java允许这个

class Status
{
    public static final Status OFF = new Status("Off");

    public static final Color defaultColor = Color.WHITE;

    Status(String name)
    {
      this(name, defaultColor);
    }
}

当然它在运行时会有问题,但Java并不关心。程序员的工作是安排init序列,编译器检查所​​有破坏的init依赖项并不容易。无论如何,问题很容易解决:

class Status
{
    // now it works, this field is initialized first
    public static final Color defaultColor = Color.WHITE;

    public static final Status OFF = new Status("Off");

但对于enum,此解决方法不适用,因为enum类型中的静态字段无法在枚举之前移动(可能出于纯语法原因)。为了避免混淆,Java为enum添加了一个额外的限制 - 无法从构造函数中引用静态字段。

这种限制是半生的。从构造函数中检查静态字段的所有可能用法并不容易(如果不是不可能)。以下代码将编译,击败限制:

enum Status
{
    OFF("Off");

    public static final Color defaultColor = Color.WHITE;
    static Color defaultColor(){ return defaultColor; }

    Status(String name)
    {
      this(name, defaultColor());
    }

答案 3 :(得分:0)