安卓中一個Activity托管的多個Fragment之間的通信

我們在寫安卓應(yīng)用時,經(jīng)常會碰到需要一個activity托管多個fragment的情況。特別是在寫平板應(yīng)用時,由于屏幕比較大,為了充分利用屏幕的空間,通常會采用左側(cè)顯示列表、右側(cè)顯示選項詳細信息的Master-Detail布局方式,如下圖所示:

Master-Detail 布局

此時必然會遇到的一個問題就是如何在左側(cè)列表Fragment和右側(cè)詳細信息Fragment之間進行通信,下面我們就來說說幾種通信方式。

1. Fragment之間直接通信(不推薦)

我們以點擊左側(cè)列表欄中的一個列表項,使得右側(cè)詳細信息欄顯示列表內(nèi)容為例。我們可以為列表項添加監(jiān)聽,在點擊一項時,左側(cè)fragment通過Fragment.getActivity().getSupportFragmentManager()調(diào)用獲取到其托管activity的FragmentManager,然后直接通過FragmentManager替換掉右側(cè)的Fragment。示例代碼如下:

public void onClick(View v) {
  Fragment fragment = CrimeFragment.newInstance(mCrime.getId());
  FragmentManager fm = getActivity().getSupportFragmentManager();
  fm.beginTransaction()
    .add(R.id.detail_fragment_container, fragment)
    .commit();
}

此方法雖然是可行的,但是它的缺點也很明顯:

  • 一個fragment直接通過托管activity的FragmentManager直接操縱其他Fragment,這就表示,fragment必須要了解activity的工作方式,而這就破壞了fragment的獨立性,使得其難以復(fù)用。
  • Fragment必須要知道托管Activity的布局文件中一些具體細節(jié),如上例中,F(xiàn)ramgnet需要知道Activity的布局中有一個id為 R.id.detail_fragment_container 組件,并且確定該組件是為右側(cè)界面預(yù)留的,這已經(jīng)大大超出了fragment的職責(zé)范圍。

2. 通過Activity使用Fragment回調(diào)接口(推薦)

為保持fragment的獨立性,我們可以在fragment中定義回調(diào)接口,委托托管activity來完成那些不應(yīng)由fragment處理的任務(wù)。托管activity將實現(xiàn)回調(diào)接口,履行托管fragment的任務(wù)。
要委托工作任務(wù)給托管activity,通常的做法是由fragment定義名為 Callbacks 的回調(diào)接口?;卣{(diào)接口定義了fragment委托給托管activity處理的工作任務(wù)。任何打算托管目標fragment的activity都必須實現(xiàn)它。

public class CrimeListFragment extends Fragment {
  ...
  private boolean mSubtitleVisible;
  private Callbacks mCallbacks;
  /**
  * Required interface for hosting activities.
  */
  public interface Callbacks {
    void onCrimeSelected(Crime crime);
  }

  ...

  @Override
  public void onAttach(Activity activity) {
    super.onAttach(activity);
    mCallbacks = (Callbacks) activity;
  }

  @Override
  public void onDetach() {
    super.onDetach();
    mCallbacks = null;
  }

  private class CrimeHolder extends RecyclerView.ViewHolder
    implements View.OnClickListener {
    ...
    @Override
    public void onClick(View v) {
      mCallbacks.onCrimeSelected(mCrime);
    }
  }
  ...
}

上面是左側(cè)列表界面的示例代碼,其中定義了一個叫Callbacks的接口,在列表項CrimeHolder被點擊時,就會調(diào)用mCallbacks.onCrimeSelected(mCrime);來調(diào)用托管activity中的對應(yīng)方法,通知其更新右側(cè)視圖。

下面我們來看看托管Activity中的代碼:

public class CrimeListActivity extends SingleFragmentActivity
    implements CrimeListFragment.Callbacks {

  @Override
  protected Fragment createFragment() {
    return new CrimeListFragment();
  }

  @Override
  protected int getLayoutResId() {
    return R.layout.activity_masterdetail;
  }

  @Overrid
  public void onCrimeSelected(Crime crime) {    
    Fragment newDetail = CrimeFragment.newInstance(crime.getId());
    getSupportFragmentManager().beginTransaction()
      .replace(R.id.detail_fragment_container, newDetail)
      .commit();
  }
  ...
}

可見,托管activity實現(xiàn)了fragment中的Callbacks接口,并在onCrimeSelected(Crime crime)方法中,根據(jù)參數(shù)crime中的信息創(chuàng)建新的右側(cè)詳細信息fragment,然后在FragmentManager中替換掉了舊的右側(cè)fragment。
通過以上方法,fragment和activity各自關(guān)心自己負責(zé)的邏輯部分,沒有破壞各自的封裝性,增強了fragment的可復(fù)用性。因此,在今后需要處理多個fragment之間的通信時,可以優(yōu)先考慮第二種實現(xiàn)方式。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容