Java switch语句:需要常量expression式,但是它是常量

所以,我正在研究这个有几个静态常量的类:

public abstract class Foo { ... public static final int BAR; public static final int BAZ; public static final int BAM; ... } 

然后,我想一个方法来获得一个相关的string基于常量:

 public static String lookup(int constant) { switch (constant) { case Foo.BAR: return "bar"; case Foo.BAZ: return "baz"; case Foo.BAM: return "bam"; default: return "unknown"; } } 

但是,当我编译时,我得到了每个3个案例标签的constant expression required错误。

我知道编译器需要在编译时知道expression式来编译开关,但为什么不是Foo.BA_常量?

我知道编译器需要在编译时知道expression式来编译开关,但为什么不是Foo.BA_常量?

虽然从字段初始化后执行的任何代码的angular度来看,它们是不变的,但它们不是JLS所要求的编译时间常量 ; 有关常量expression式所需的定义,请参见§15.28常量expression式。 这涉及到§4.12.4最终variables ,它定义了一个“常量variables”,如下所示:

我们调用一个原始types或types为String的variables,该variables是最终的,并用一个编译时常量expression式(§15.28)作为常量variables进行初始化。 variables是否为常量variables可能对类初始化(§12.4.1),二进制兼容性(§13.1,§13.4.9)和明确赋值(§16)有影响。

在你的例子中,Foo.BA *variables没有初始值设定项,因此没有被定义为“常量variables”。 修复很简单, 将Foo.BA *variables声明更改为具有编译时常量expression式的初始化程序。

你得到所需的常量expression,因为你离开常量的值。 尝试:

 public abstract class Foo { ... public static final int BAR=0; public static final int BAZ=1; public static final int BAM=2; ... } 

因为那些不是编译时间常量。 考虑下面的有效代码:

 public static final int BAR = new Random().nextInt(); 

您只能在运行时知道BAR的值。

我在Android上遇到这个错误,而我的解决scheme只是使用:

 public static final int TAKE_PICTURE = 1; 

代替

 public static int TAKE_PICTURE = 1; 

你可以在这个例子中使用一个枚举:

 public class MainClass { enum Choice { Choice1, Choice2, Choice3 } public static void main(String[] args) { Choice ch = Choice.Choice1; switch(ch) { case Choice1: System.out.println("Choice1 selected"); break; case Choice2: System.out.println("Choice2 selected"); break; case Choice3: System.out.println("Choice3 selected"); break; } } } 

源: 使用枚举切换语句

我build议你使用枚举:)

看一下这个:

 public enum Foo { BAR("bar"), BAZ("baz"), BAM("bam"); private final String description; private Foo(String description) { this.description = description; } public String getDescription() { return description; } } 

那么你可以像这样使用它:

 System.out.println(Foo.BAR.getDescription());