Android庫項(xiàng)目中的資源ID沖突的解決方法
1、前言
Android Studio對模塊化開發(fā)提供的一個很有用的功能就是可以在主項(xiàng)目下新建庫項(xiàng)目(Module),但是在使用庫項(xiàng)目時卻有一個問題就是資源ID沖突,因?yàn)榫幾g時SDK會自動幫我們處理這個問題,所以一般我們不會察覺到,但是在某些情況下,我們需要意識到這個問題的存在。
比如,在新建的庫項(xiàng)目中使用如下代碼:
public void onButtonClick(View view) { switch (view.getId()) { case R.id.button_1: break; case R.id.button_2; break; } }
IDE會提示:
Resource IDs cannot be used in a switch statement in Android library modules less.
Validates using resource IDs in a switch statement in Android library module. Resource IDs are non final in the library projects since SDK tools r14, means that the library code cannot treat these IDs as constants.
再比如,我們在庫項(xiàng)目中以如下方式使用ButterKnife,編譯時就會報錯。
@OnClick(R.id.button_1) public void onButtonClick(View view) { }
2、分析
無論是 switch 語句還是注解,都有一個要求就是使用的值必須是常量。在主項(xiàng)目中, R類中的成員變量都被 static final 修飾,而在庫項(xiàng)目中僅被 static 修飾。
// 庫項(xiàng)目中生成的R類: public final class R { public static final class id { public static int button_1 = 0x7f0c0001; } } // 主項(xiàng)目中生成的R類: public final class R { public static final class id { public static final int text_1 = 2131165184; } }
為什么庫項(xiàng)目中生成的資源ID沒有被 final 修飾呢?官方解釋如下:
Non-constant Fields in Case Labels
當(dāng)多個庫項(xiàng)目進(jìn)行合并時,不同項(xiàng)目中的資源ID可能會重復(fù)。在ADT 14之前,無論是主項(xiàng)目還是庫項(xiàng)目,資源ID統(tǒng)一被定義為 final 類型的靜態(tài)變量。這樣照成的結(jié)果就是主項(xiàng)目進(jìn)行編譯時一旦發(fā)現(xiàn)資源ID沖突,庫項(xiàng)目中對應(yīng)的資源文件以及引用資源文件的代碼都需要重新編譯。
如果代碼中使用了被 static final 修飾的變量,那這個變量實(shí)際上就是一個常量,編譯時會直接使用它的值進(jìn)行替換。在編譯時,如果庫項(xiàng)目與主項(xiàng)目的資源ID發(fā)生了重復(fù),資源被分配了新的ID后庫項(xiàng)目之前編譯過的代碼也就失效了。
那么當(dāng)庫項(xiàng)目R類中的變量僅被 static 修飾后會起到什么作用呢,我們可以看一下編譯后的字節(jié)碼再反編譯后的樣子。
// 主項(xiàng)目中的Activity: public class MainActivity extends AppCompatActivity { protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // 源代碼:setContentView(R.layout.activity_main); this.setContentView(2131296283); } } // 庫項(xiàng)目中的Activity: public LibActivity extends AppCompatActivity { protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); this.setContentView(R.layout.activity_lib); } }
主項(xiàng)目R類中的資源ID被 static final 修飾,編譯時直接被替換成了對應(yīng)的常量。庫項(xiàng)目R類中的資源ID僅被 static 修飾,所以保留了變量。這樣當(dāng)資源ID發(fā)送沖突時,主項(xiàng)目R類不變,修改庫項(xiàng)目R類中的變量,庫項(xiàng)目已經(jīng)編譯過的代碼仍有效。
3、ButterKnife中的R2類
既然庫項(xiàng)目中的資源ID不可以定義為常量,那如何在庫文項(xiàng)目使用ButterKnife呢,作者提供了R2類供我使用。
@OnClick({R2.id.button_1, R2.id.button_2}) public void onButtonClick(View view) { int id = view.getId(); if (id == R.id.button_1) { // ... } else if (id == R.id.button_2) { // ... } }
沒錯在注解中使用R2類,但是在代碼里還是需要使用R類,因?yàn)镽類中的ID不是常量,所以只能使用 if 語句進(jìn)行判斷。
先來看一下ButterKnife為我們生成的R2類與R類有什么不同:
// 庫項(xiàng)目中的R類: public final class R { public static final class id { public static int button_1 = 0x7f0c0001; } } // 庫項(xiàng)目中ButterKnife為我們生成的R2類: public final class R2 { public static final class id { public static final int button_1 = 0x7f0c0001; } }
ButterKnife做的工作很簡單,僅僅是把R類中的變量搬到了R2類里,然后給所有的變量都加上了 final 。根據(jù)前面所說,當(dāng)項(xiàng)目整體編譯時,庫項(xiàng)目的資源ID一旦與主項(xiàng)目的資源ID發(fā)送沖突,庫項(xiàng)目的資源會被重新分配ID導(dǎo)致其R類被修改。顯然這個過程并不涉及R2類,R2類中保留的仍然是過時的ID。但是ButterKnife提供的注解的作用是什么,它們并不是為了提供運(yùn)行時信息,而是為了在編譯時生成代碼。
public class LibActivity_ViewBinding implements Unbinder { private LibActivity target; private View view_button_1; private View view_button_2; @UiThread public LibActivity_ViewBinding(final LibActivity target, View source) { this.target = target; View view = Utils.findRequiredView(source, R.id.button_1, "method 'onButtonClick'"); this.view_button_1 = view; //view.setOnClickListener.... view = Utils.findRequiredView(source, R.id.button_2, "method 'onButtonClick'"); this.view_button_2 = view; //view.setOnClickListener.... } }
在ButterKnife生成的代碼中,使用的仍然是R類。R2起到的作用僅僅是提供一個符號名,只要讓程序知道在生成代碼時對應(yīng)哪一個變量即可。這個方法可以說是很“tricky”了。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
詳解Android沉浸式實(shí)現(xiàn)兼容解決辦法
本篇文章主要介紹了詳解Android沉浸式實(shí)現(xiàn)兼容解決辦法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-11-11Android編程開發(fā)之NotiFication用法詳解
這篇文章主要介紹了Android編程開發(fā)之NotiFication用法,結(jié)合實(shí)例形式較為詳細(xì)的分析了NotiFication的功能、使用技巧與注意事項(xiàng),需要的朋友可以參考下2015-12-12Android自定義View仿大眾點(diǎn)評星星評分控件
這篇文章主要為大家詳細(xì)介紹了Android自定義View仿大眾點(diǎn)評星星評分控件,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-03-03Android Studio多工程引用同一個library項(xiàng)目配置的解決方法
大家在使用android studio的時候,會遇到多個項(xiàng)目引用相同的library這篇文章主要介紹了Android Studio多工程引用同一個library項(xiàng)目配置方法,需要的朋友可以參考下2018-03-03Android xUtils更新到3.0后的基本使用規(guī)則詳解
xUtils是基于android的開發(fā)框架,簡化了很多的開發(fā)步驟,可以說是非常好的開發(fā)工具。下面小編給大家?guī)砹薃ndroid xUtils更新到3.0后的基本使用規(guī)則詳解,感興趣的朋友一起學(xué)習(xí)吧2016-08-08Android應(yīng)用隱私合規(guī)檢測實(shí)現(xiàn)方案詳解
這篇文章主要介紹了Android應(yīng)用隱私合規(guī)檢測實(shí)現(xiàn)方案,我們需要做的就是提前檢測好自己的應(yīng)用是否存在隱私合規(guī)問題,及時整改過來,下面提供Xposed Hook思路去檢測隱私合規(guī)問題,建議有Xposed基礎(chǔ)的童鞋閱讀,需要的朋友可以參考下2022-07-07詳細(xì)講解Android中使用LoaderManager加載數(shù)據(jù)的方法
這篇文章主要介紹了Android中使用LoaderManager加載數(shù)據(jù)的方法,講到了LoaderManager的異步加載與聲明周期的管理等相關(guān)用法,需要的朋友可以參考下2016-04-04