關于R.id.XX是否可以作為Case的鍵值棍丐?
(2015-05-26 10:25:12
移植一個項目到框架中误辑,原項目代碼中使用:
swich(id)
case R.id.XX:break;
case R.id.XX:break;
代碼不報錯,但我移植到我的Frame框架中(Frame框架是作為庫存在的)報錯歌逢,提示case鍵必須使用常量巾钉。
開始想不明白,為什么兩個項目中會有如此區(qū)別呢秘案?后來看了一篇博客砰苍,才恍然大悟潦匈。
Case 標簽中的常量字段——在正常的 Android 項目中,資源R類里的常量是這樣聲明的:
public static final int main=0x7f030004;
然而赚导,到 ADT 14為止茬缩,在類庫項目中,他們會被這樣聲明:
public static int main=0x7f030004;
換句話說吼旧,在庫項目中這些常量都不是final的凰锡。原因很簡單:當多個庫項目整合時,字段的實際值(必須是唯一的)可能會出現(xiàn)碰撞圈暗。在ADT 14 之前掂为,所有這樣的字段都是final的,所以導致所有的庫被使用的時候员串,主項目必須編譯他們的所有資源和相關 Java 代碼勇哗。這樣的性能是很差的,因為它使構建的速度非常慢寸齐。它還阻礙了不包含源代碼的庫項目的分發(fā)欲诺,限制庫項目的使用范圍。
字段不再是 final 的原因是意味著庫jar包可以只編譯一次渺鹦,并且能直接在其他項目中復用瞧栗。以及允許分發(fā)庫項目的二進制版本(將在 r15出現(xiàn)),這使得構建更快海铆。
然而,它對庫的源碼也有一個影響挣惰。以下形式的代碼將不再能夠編譯:
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 語句轉(zhuǎn)換成 if-else 語句珍语。幸運的是,這在 Eclipse 中操作起來很簡單竖幔。只需要將插入符號放在switch關鍵字上板乙,并按 Ctrl+1 (在Mac上是 Cmd+1):
在上述場景中,它將把switch語句轉(zhuǎn)換成如下: 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 代碼中拳氢,并且它對性能的影響可以忽略不計募逞。
我們有一個檢測器,可以發(fā)現(xiàn)這些錯誤 (引用R字段的非常量 case 標簽)馋评,并提供問題的簡要說明 (并指向此頁面以獲取詳細信息放接。) 有關自動檢測的詳細信息。
P.S. 如果你的 switch 語句如下所示:
switch (view.getId()) {
那么結(jié)果會轉(zhuǎn)換成每一個if檢查都重復調(diào)用 view.getId() 的效率低下的if/else鏈留特。你需要先提取此表達式 (使用“提取本地變量(Extract Local Variable)”的重構快捷鍵鍵)纠脾,然后再轉(zhuǎn)換這條 switch 語句玛瘸。