溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Android庫項(xiàng)目中的資源ID沖突的解決方法

發(fā)布時(shí)間:2020-09-10 02:47:19 來源:腳本之家 閱讀:403 作者:mmmmar 欄目:移動(dòng)開發(fā)

1、前言

Android Studio對模塊化開發(fā)提供的一個(gè)很有用的功能就是可以在主項(xiàng)目下新建庫項(xiàng)目(Module),但是在使用庫項(xiàng)目時(shí)卻有一個(gè)問題就是資源ID沖突,因?yàn)榫幾g時(shí)SDK會自動(dòng)幫我們處理這個(gè)問題,所以一般我們不會察覺到,但是在某些情況下,我們需要意識到這個(gè)問題的存在。

比如,在新建的庫項(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,編譯時(shí)就會報(bào)錯(cuò)。

@OnClick(R.id.button_1)
public void onButtonClick(View view) { 

} 

 2、分析

無論是 switch 語句還是注解,都有一個(gè)要求就是使用的值必須是常量。在主項(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)多個(gè)庫項(xiàng)目進(jìn)行合并時(shí),不同項(xiàng)目中的資源ID可能會重復(fù)。在ADT 14之前,無論是主項(xiàng)目還是庫項(xiàng)目,資源ID統(tǒng)一被定義為 final 類型的靜態(tài)變量。這樣照成的結(jié)果就是主項(xiàng)目進(jìn)行編譯時(shí)一旦發(fā)現(xiàn)資源ID沖突,庫項(xiàng)目中對應(yīng)的資源文件以及引用資源文件的代碼都需要重新編譯。

如果代碼中使用了被 static final 修飾的變量,那這個(gè)變量實(shí)際上就是一個(gè)常量,編譯時(shí)會直接使用它的值進(jìn)行替換。在編譯時(shí),如果庫項(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 修飾,編譯時(shí)直接被替換成了對應(yīng)的常量。庫項(xiàng)目R類中的資源ID僅被 static 修飾,所以保留了變量。這樣當(dāng)資源ID發(fā)送沖突時(shí),主項(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) {
    // ...
  }
} 

沒錯(cuò)在注解中使用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)目整體編譯時(shí),庫項(xiàng)目的資源ID一旦與主項(xiàng)目的資源ID發(fā)送沖突,庫項(xiàng)目的資源會被重新分配ID導(dǎo)致其R類被修改。顯然這個(gè)過程并不涉及R2類,R2類中保留的仍然是過時(shí)的ID。但是ButterKnife提供的注解的作用是什么,它們并不是為了提供運(yùn)行時(shí)信息,而是為了在編譯時(shí)生成代碼。

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起到的作用僅僅是提供一個(gè)符號名,只要讓程序知道在生成代碼時(shí)對應(yīng)哪一個(gè)變量即可。這個(gè)方法可以說是很“tricky”了。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI