在正常的 Android 项目中,资源R类里的常量是这样声明的:
public static final int main=0x7f030004;
然而,到 ADT 14为止,在类库项目中,他们会被这样声明:
public static int main=0x7f030004;
换句话说,在库项目中这些常量都不是final的。原因很简单:当多个库项目整合时,字段的实际值(必须是唯一的)可能会出现碰撞。在ADT 14 之前,所有这样的字段都是final的,所以导致所有的库被使用的时候,主项目必须编译他们的所有资源和相关 Java 代码。这样的性能是很差的,因为它使构建的速度非常慢。它还阻碍了不包含源代码的库项目的分发,限制库项目的使用范围。
字段不再是 final 的原因是意味着库jar包可以只编译一次,并且能直接在其他项目中复用。以及允许分发库项目的二进制版本(将在 r15出现),这使得构建更快。
然而,它对库的源码也有一个影响。以下形式的代码将不再能够编译:
然而,它对库的源码也有一个影响。以下形式的代码将不再能够编译:
int id = view.getId();
switch (id) {
case R.id.button1: action1();
break;
case R.id.button2:
action2();
break;
case R.id.button3:
action3();
break;
}
这是因为
switch
语句需要所有case的标签,如R.id.button1
,在编译时是个常量 (这样值可以直接复制到.class 文件中)。为此解决方案很简单: 将 switch 语句转换成 if-else 语句。幸运的是,这在 Eclipse 中操作起来很简单。只需要将插入符号放在switch关键字上,并按 Ctrl+1 (在Mac上是 Cmd+1):
在上述场景中,它将把
switch
语句转换成如下:int id = view.getId();
if (id == R.id.button1) {
action1();
} else if (id == R.id.button2) {
action2();
} else if (id == R.id.button3) {
action3();
}
这通常是在 UI 代码中,并且它对性能的影响可以忽略不计。
我们有一个检测器,可以发现这些错误 (引用R字段的非常量 case 标签),并提供问题的简要说明 (并指向此页面以获取详细信息。)
P.S. 如果你的 switch 语句如下所示:
switch (view.getId()) {
那么结果会转换成每一个
if
检查都重复调用 view.getId() 的效率低下的if/else
链。你需要先提取此表达式 (使用“提取本地变量(Extract Local Variable)”的重构快捷键键),然后再转换这条 switch 语句。