Android中Fragment與Activity之間的互動(兩種實現方式),androidfragment
(未給Fragment的布局設定BackGound)
之前關於Android中Fragment的概念以及建立方式,我專門寫了一篇博文《Android中Fragment的兩種建立方式》,就如何建立Fragment混合布局做了詳細的分析,今天就來詳細說道說道Fragment與宿主Activity之間是如何?資料互動的。
我們可以這樣理解,宿主Activity中的Fragment之間要實現資訊互動,就必須通過宿主Activity,Fragment之間是不可能直接實現資訊互動的。
Fragment與Fragment或者說與Activity(將部分Fragment包含的的布局直接部署在Activity中)的資料互動我個人總結了兩種實現方式:
(1)通過Bundle傳參,從而實現Fragment之間的資料互動
(2)通過在Fragment內部定義一個回調介面,並需要宿主Activity實現它。當Activity通過介面接收到回調時,可以在必要時與布局中的其它Fagment共用資訊。
在開始兩種實現方式之前,先向大家介紹並示範其中一些個人化參數不同設定後的效果,之前向大家介紹過要管理宿主Activity中的Fragment,必須得到FragmentManager(片段管理),而FragmentManager要實現對Fragment的增刪改換等操作(事務),則必須啟用FragmentTransaction,這裡主要向大家示範添加FragmentTransaction.addToBackStack方法前後以及在未給Fragment設定BackGround的情況下使用FragmentTransaction.add與replace的不同效果,更加詳細的方法介紹以及用法請大家參照API詳細瞭解。
未設定
FragmentTransaction.addToBackStack方法示範效果(開啟多層後,按回退,直接退出程式):
設定FragmentTransaction.addToBackStack方法示範效果(這裡沒有設定監聽後台棧變化的監聽器進行判斷處理):
貌似看不出差別,其實在按回退鍵時,這個是根據開啟的順序,逐個退出
未給Fragment設定BackGround的情況下使用FragmentTransaction.add的示範效果:注意啦注意啦,這裡著重聲明的是未給右側Fragment的布局設定BackGound的情況,如果設定的BackGound,那麼實現效果和replace沒有差別,這也是今天逾時這麼多的主要原因
不同的實現效果大家都看到了,我們開始示範實現代碼:
我們使用的是Android中Fragment的兩種建立方式中(通過java代碼將fragment添加到宿主Activity中)的布局檔案,布局檔案代碼請參考http://www.cnblogs.com/panhouye/p/6185093.html
(1)通過Bundle傳參
第一步:
右側Fragment對應的java代碼RightFragment.java:
1 import android.app.Fragment; 2 import android.os.Bundle; 3 import android.view.LayoutInflater; 4 import android.view.View; 5 import android.view.ViewGroup; 6 import android.widget.TextView; 7 /** 8 * Created by panchengjia on 2016/12/18. 9 */10 public class RightFragment extends Fragment {11 public RightFragment() {12 }13 /*Fragment的傳參方式(通過Bundle對象來傳遞)14 *採用這種傳參方式可以保證使用者在橫豎屏切換時所15 * 傳遞的參數不會丟失16 */17 public static RightFragment getInstance(String data){18 RightFragment rightFragment = new RightFragment();19 Bundle bundle = new Bundle();20 //將需要傳遞的字串以索引值對的形式傳入bundle21 bundle.putString("data",data);22 rightFragment.setArguments(bundle);23 return rightFragment;24 }25 @Override26 public void onCreate(Bundle savedInstanceState) {27 super.onCreate(savedInstanceState);28 }29 @Override30 public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {31 View view =inflater.inflate(R.layout.right_layout,container,false);32 TextView tv = (TextView) view.findViewById(R.id.tv);33 String data = getArguments().getString("data");34 tv.setText(data);35 return view;36 }37 @Override38 public void onPause() {39 super.onPause();40 }41 }
第二步:
主布局宿主Activity對應的java實現代碼MainActivity.java:
1 import android.app.FragmentManager; 2 import android.app.FragmentTransaction; 3 import android.support.v7.app.AppCompatActivity; 4 import android.os.Bundle; 5 import android.view.View; 6 import android.widget.Button; 7 public class Main3Activity extends AppCompatActivity { 8 FragmentManager fragmentManager; 9 FragmentTransaction fragmentTransaction;10 LeftFragment leftFragment;11 Button panhouye,bikonghai;//聲明leftfragment中的按鈕12 @Override13 protected void onCreate(Bundle savedInstanceState) {14 super.onCreate(savedInstanceState);15 setContentView(R.layout.activity_main2);16 //擷取fragmentManager17 fragmentManager=getFragmentManager();18 //通過findFragmentById找到leftFragment19 leftFragment = (LeftFragment) fragmentManager.findFragmentById(R.id.left);20 //找到對應的導覽按鈕並設定點擊事件21 panhouye = (Button) leftFragment.getView().findViewById(R.id.panhouye);22 bikonghai = (Button) leftFragment.getView().findViewById(R.id.bikonghai);23 panhouye.setOnClickListener(new View.OnClickListener() {24 @Override25 public void onClick(View v) {26 //調用方法修改rightfragment中的常值內容27 switchButton("我是潘侯爺");28 }29 });30 bikonghai.setOnClickListener(new View.OnClickListener() {31 @Override32 public void onClick(View v) {33 switchButton("我是碧空海");34 }35 });36 //設定開啟Activity後rightfragment中預設的常值內容37 switchButton("我是潘侯爺");38 }39 //定義方法填充Activity右側的fragment,並通過傳參修改常值內容40 public void switchButton(String data){41 fragmentManager=getFragmentManager();42 fragmentTransaction=fragmentManager.beginTransaction();43 //通過調用RightFragment中的getInstance方法傳修改文本44 RightFragment rightFragment =RightFragment.getInstance(data);45 //此時使用add方法會造成右側fragment中文本重疊(未設定BackGround時)46 fragmentTransaction.replace(R.id.right,rightFragment);47 fragmentTransaction.commit();48 }49 }
(2)介面回調第一步:
左側fragment的java實現代碼LeftFragment.java檔案
本次示範通過點擊左側Fragment中的按鈕點擊觸發與右側Fragment的資料互動,所以需在本類中添加回調介面用於在宿主Activity中回調修改右側文本的方法。
1 import android.app.Fragment; 2 import android.os.Bundle; 3 import android.view.LayoutInflater; 4 import android.view.View; 5 import android.view.ViewGroup; 6 import android.widget.Button; 7 /** 8 * Created by panchengjia on 2016/12/18. 9 */10 public class LeftFragment extends Fragment implements View.OnClickListener{11 //聲明內部定義的回調介面12 CallBackListener callBackListener;13 //聲明布局中事件觸發按鈕14 Button panhouye,bikonghai;15 @Override16 public void onCreate(Bundle savedInstanceState) {17 super.onCreate(savedInstanceState);18 //通過getActivity()擷取用於回調修改文本方法的介面19 callBackListener= (CallBackListener) getActivity();20 }21 @Override22 public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {23 View view =inflater.inflate(R.layout.left_layout,container,false);24 panhouye = (Button) view.findViewById(R.id.panhouye);25 bikonghai= (Button) view.findViewById(R.id.bikonghai);26 panhouye.setOnClickListener(this);//為按鈕設定監聽事件27 bikonghai.setOnClickListener(this);28 return view;29 }30 @Override31 public void onPause() {32 super.onPause();33 }34 35 @Override36 public void onClick(View v) {37 switch (v.getId()){38 case R.id.panhouye:39 callBackListener.setText("我是潘侯爺");40 break;41 case R.id.bikonghai:42 callBackListener.setText("我是碧空海");43 break;44 }45 }46 //設定用於修改文本的回調介面47 public static interface CallBackListener{48 public void setText(String data);49 }50 }
第二步:
右側fragment的java實現代碼RightFragment.java檔案
本次示範右側為文本顯示fragment,也是點擊左側的按鈕後,通過改變文本的形式體現點擊事件的處理,所以必須在本Fragment類中添加文本修改的方法。
1 import android.app.Fragment; 2 import android.os.Bundle; 3 import android.support.annotation.Nullable; 4 import android.view.LayoutInflater; 5 import android.view.View; 6 import android.view.ViewGroup; 7 import android.widget.TextView; 8 /** 9 * Created by panchengjia on 2016/12/18.10 */11 public class RightFragment extends Fragment {12 //聲明fragment中的TextView,用於建立修改文本的方法13 private TextView tv;14 @Override15 public void onCreate(Bundle savedInstanceState) {16 super.onCreate(savedInstanceState);17 }18 @Override19 public void onPause() {20 super.onPause();21 }22 @Nullable23 @Override24 public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {25 View view =inflater.inflate(R.layout.right_layout,container,false);26 tv = (TextView) view.findViewById(R.id.tv);27 return view;28 }29 //在這裡設定修改自身文本的方法30 public void setFragmentText(String data){31 tv.setText(data);32 }33 }
第三步:
主介面宿主Acvtivity的java實現代碼MainActivity.java檔案(為了接收Fragment事件回調,宿主的Activity必須實現回調介面):
1 import android.app.FragmentManager; 2 import android.app.FragmentTransaction; 3 import android.support.v7.app.AppCompatActivity; 4 import android.os.Bundle; 5 public class MainActivity extends AppCompatActivity implements LeftFragment.CallBackListener{ 6 FragmentManager fragmentManager; 7 FragmentTransaction fragmentTransaction; 8 /*leftfragment已經在主布局檔案中聲明, 9 *這裡僅需要通過代碼聲明部署rightFragment10 */11 RightFragment rightFragment;12 @Override13 protected void onCreate(Bundle savedInstanceState) {14 super.onCreate(savedInstanceState);15 setContentView(R.layout.activity_main);16 //初始化主布局(主要目的是為主布局填充fragments)17 initActivity();18 }19 private void initActivity() {20 fragmentManager = getFragmentManager();21 fragmentTransaction = fragmentManager.beginTransaction();22 rightFragment = new RightFragment();23 fragmentTransaction.add(R.id.right,rightFragment);24 fragmentTransaction.commit();25 }26 //介面實現方法,用於回調RightFragment類中定義的修改文本的方法27 @Override28 public void setText(String data) {29 rightFragment.setFragmentText(data);30 }31 }
小結:
本次示範介面回調的實現方式看起來比使用bundle傳參的代碼量大了一些,但在實際開發中,我們面臨的Fragment不僅僅是眼前的這兩個,而使用介面回調實現互動資料的方法能更好的實現重用Fragment UI組件,從根本上解決的大量代碼重用的問題,建議大家熟練掌握介面回調來實現資料互動。