19

我想使用 javaswitch语句,它使用class名称作为case常量。有可能吗?还是我必须复制类名?

由于编译器错误,以下代码不起作用:

case 表达式必须是常量表达式

String tableName = "MyClass1";

...

switch (tableName) {
case MyClass1.class.getSimpleName():
    return 1;
case MyClass2.class.getSimpleName():
    return 2;
default:
    return Integer.MAX_VALUE;
}

这是该问题的在线演示(openjdk 1.8.0_45):http: //goo.gl/KvsR6u

4

3 回答 3

13

编译器错误已经说明了。case 标签必须是常量表达式,类文字和调用getSimpleName()它们的结果都不是常量表达式。

一个可行的解决方案是:

String tableName = "MyClass1";
...
switch (tableName) {
    case "MyClass1":
        return 1;
    case "MyClass2":
        return 2;
    default:
        return Integer.MAX_VALUE;
}

表达式MyClass1.class.getSimpleName()并不比 更简单"MyClass1",但是,当然,不会有任何编译时检查名称是否与现有类匹配,并且重构工具或混淆器不会注意到类MyClass1和字符串文字之间的关系"MyClass1"

没有解决办法。减少问题的唯一方法是在关联类中声明键以记录关系,例如

class MyClass1 {
    static final String IDENTIFIER = "MyClass1";
    ...
}
class MyClass2 {
    static final String IDENTIFIER = "MyClass2";
    ...
}
...
String tableName = MyClass1.IDENTIFIER;
...
switch (tableName) {
    case MyClass1.IDENTIFIER:
        return 1;
    case MyClass2.IDENTIFIER:
        return 2;
    default:
        return Integer.MAX_VALUE;
}

这记录了与读者的关系,但工具仍然不能确保实际的字符串内容与类名匹配。但是,取决于您想要实现的目标,现在可能变得无关紧要,字符串内容是否与类名匹配......</p>

于 2016-01-05T15:59:05.203 回答
8

与其使用开关,不如将映射存储在地图中?

创建一个 String 到 Integer 的映射,并将所有类名映射到它们的返回值。

在请求中,如果条目不存在,则返回默认值。否则,返回映射中的值。

于 2016-01-05T13:54:30.570 回答
3

而不是 Switch..case 为什么不使用 If..Else。在我知道之前应该在所有版本的java中工作。

if (tableName.equals(MyClass1.class.getSimpleName())) {
     return 1;
} else if (tableName.equals(MyClass2.class.getSimpleName())) {
     return 2;
} else {
     return Integer.MAX_VALUE;
}
于 2016-01-05T14:08:20.750 回答