編輯:Android資訊
Fragment是Android中的重要組件,在Android 3.0的時候添加進來。
關於Fragment的生命周期,我相信了解過的開發人員都應該把以下方法脫口而出:onAttach, onCreate, onCreateView, onViewCreated, onActivityCreated, onStart, onResume, onPause, onStop, onDestroyView, onDestroy, onDetach.
當Fragment以靜態的方式,即通過在布局文件中以其它控件的方式設置時,它的生命周期隨所在Activity的生命周期而發生變化。此時其生命周期的方法調用過程是這樣的:
1,當首次展示布局頁面時,其生命周期方法調用的順序是:
2,而當關閉手機屏幕或者手機屏幕變暗時,其其生命周期方法調用的順序是:
3,當再次對手機屏幕解鎖或者手機屏幕變亮時,其生命周期方法調用的順序是:
4,而當對當前Fragment所在屏幕按返回鍵時,其生命周期方法調用的順序是:
1 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onPause 2 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onStop 3 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onDestroyView 4 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onDestroy 5 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onDetach
但是當使用FragmentManager動態的管理Fragment並且涉及到是否addToBackStack時,其生命周期的展現就略微顯得有些復雜了。但是還沒有復雜到無法理解。
好,下面,我們就探究一下這些問題。
首先,我們重寫了兩個Fragment,主要是重寫了它們的生命周期方法,通過在其生命周期方法中打印出Log的方式來顯示其方法的調用。
兩個類分別是:
package com.yeepay.fraglifecircletest.frag; import android.app.Activity; import android.app.Fragment; import android.os.Bundle; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import com.yeepay.fraglifecircletest.R; public class FragA extends Fragment { private static final String TAG = FragA.class.getSimpleName(); @Override public void onAttach(Activity activity) { super.onAttach(activity); Log.i(TAG, "onAttach"); } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Log.i(TAG, "onCreate"); } @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { Log.i(TAG, "onCreateView"); return inflater.inflate(R.layout.fragment_test_a, null, false); } @Override public void onViewCreated(View view, Bundle savedInstanceState) { Log.i(TAG, "onViewCreated"); super.onViewCreated(view, savedInstanceState); } @Override public void onDestroy() { Log.i(TAG, "onDestroy"); super.onDestroy(); } @Override public void onDetach() { Log.i(TAG, "onDetach"); super.onDetach(); } @Override public void onDestroyView() { Log.i(TAG, "onDestroyView"); super.onDestroyView(); } @Override public void onStart() { Log.i(TAG, "onStart"); super.onStart(); } @Override public void onStop() { Log.i(TAG, "onStop"); super.onStop(); } @Override public void onResume() { Log.i(TAG, "onResume"); super.onResume(); } @Override public void onPause() { Log.i(TAG, "onPause"); super.onPause(); } @Override public void onActivityCreated(Bundle savedInstanceState) { Log.i(TAG, "onActivityCreated"); super.onActivityCreated(savedInstanceState); } } FragA.java
package com.yeepay.fraglifecircletest.frag; import android.app.Activity; import android.app.Fragment; import android.os.Bundle; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import com.yeepay.fraglifecircletest.R; public class FragB extends Fragment { private static final String TAG = FragB.class.getSimpleName(); @Override public void onAttach(Activity activity) { super.onAttach(activity); Log.i(TAG, "onAttach"); } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Log.i(TAG, "onCreate"); } @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { Log.i(TAG, "onCreateView"); return inflater.inflate(R.layout.fragment_test_b, null, false); } @Override public void onViewCreated(View view, Bundle savedInstanceState) { Log.i(TAG, "onViewCreated"); super.onViewCreated(view, savedInstanceState); } @Override public void onDestroy() { Log.i(TAG, "onDestroy"); super.onDestroy(); } @Override public void onDetach() { Log.i(TAG, "onDetach"); super.onDetach(); } @Override public void onDestroyView() { Log.i(TAG, "onDestroyView"); super.onDestroyView(); } @Override public void onStart() { Log.i(TAG, "onStart"); super.onStart(); } @Override public void onStop() { Log.i(TAG, "onStop"); super.onStop(); } @Override public void onResume() { Log.i(TAG, "onResume"); super.onResume(); } @Override public void onPause() { Log.i(TAG, "onPause"); super.onPause(); } @Override public void onActivityCreated(Bundle savedInstanceState) { Log.i(TAG, "onActivityCreated"); super.onActivityCreated(savedInstanceState); } } FragB.java
1,當我們通過以下方式添加FragA時,
1 FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction(); 2 fragA = new FragA(); 3 fragmentTransaction.replace(R.id.frag_container, fragA, fragNames[0]); 4 fragmentTransaction.commit();
它的生命周期展示方式是同在布局文件中靜態設置的表現一模一樣的,這裡不再詳細展開,大家可以查看一下以上內容。
2,當我們以如下方式展示FragA並且沒有addToBackStack時,
@Override public void onClick(View v) { FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction(); switch (v.getId()) { case R.id.button1: if (fragA == null) { fragA = new FragA(); fragmentTransaction.replace(R.id.frag_container, fragA, fragNames[0]); // fragmentTransaction.addToBackStack(fragNames[0]); } else { Fragment fragment = fragmentManager.findFragmentByTag(fragNames[0]); fragmentTransaction.replace(R.id.frag_container, fragment, fragNames[0]); } break; case R.id.button2: if (fragB == null) { fragB = new FragB(); fragmentTransaction.replace(R.id.frag_container, fragB, fragNames[1]); // fragmentTransaction.addToBackStack(fragNames[1]); } else { Fragment fragment = fragmentManager.findFragmentByTag(fragNames[1]); fragmentTransaction.replace(R.id.frag_container, fragment, fragNames[1]); } break; default: break; } fragmentTransaction.commit(); }
FragA生命周期調用順序是:
此時,如果再點擊另外一個按鈕B,將FragB展示出來,FragA和FragB的生命周期展示方式是:
可以看到,FragA調用順序為onPause, onStop, onDestroyView, onDestroy, onDetach.這說明,FragA已經被FragmentManager完全拋棄了,取而代之的是FragB的完全展現。而如果此時按返回鍵的話,FragB的生命周期也將是onPause, onStop, onDestroyView, onDestroy, onDetach。這說明,在添加Fragment時如果沒有調用addToBackStack方式的話,當FragmentManager更換Fragment時,是不保存Fragment的狀態的。
3,下面我們在替換Fragment時順便addToBackStack,則其生命周期展現方式是:
replace FragA and addToBackStack ######################################################################################## 01-13 17:08:43.359 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onAttach 01-13 17:08:43.359 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onCreate 01-13 17:08:43.359 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onCreateView 01-13 17:08:43.359 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onViewCreated 01-13 17:08:43.359 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onActivityCreated 01-13 17:08:43.359 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onStart 01-13 17:08:43.359 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onResume
可以看得出來,此時的生命周期方法調用是跟沒有addToBackStack時沒有任何區別的。
然後通過點擊按鈕B,使用FragB來替換FragA,此時FragA和FragB的生命周期方法調用順序是:
and then replace FragB and addToBackStack &&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onPause 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onStop 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onDestroyView 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragB﹕ onAttach 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragB﹕ onCreate 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragB﹕ onCreateView 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragB﹕ onViewCreated 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragB﹕ onActivityCreated 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragB﹕ onStart 01-13 17:08:46.959 3102-3102/com.yeepay.fraglifecircletest I/FragB﹕ onResume
由此可以看出,FragA生命周期方法只是調用到了onDestroyView,而onDestroy和onDetach則沒有被調用,這說明FragA的界面已經被銷毀了,但是FragmentManager並沒有完全銷毀FragA,FragA依然有狀態保存在FragmentManager裡面。
然後再點擊按鈕A,使用FragA來替換當前顯示的FragB,此時FragA和FragB的生命周期方法調用順序為:
and then replace FragA again &&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& 01-13 17:08:51.869 3102-3102/com.yeepay.fraglifecircletest I/FragB﹕ onPause 01-13 17:08:51.869 3102-3102/com.yeepay.fraglifecircletest I/FragB﹕ onStop 01-13 17:08:51.869 3102-3102/com.yeepay.fraglifecircletest I/FragB﹕ onDestroyView 01-13 17:08:51.869 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onCreateView 01-13 17:08:51.869 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onViewCreated 01-13 17:08:51.869 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onActivityCreated 01-13 17:08:51.869 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onStart 01-13 17:08:51.869 3102-3102/com.yeepay.fraglifecircletest I/FragA﹕ onResume
可以看到,FragB的生命方法調用順序是跟FragB替換FragA時FragA的調用順序一致的,作用就是只銷毀了視圖,但是依然保留了Fragment的狀態。而此時FragA的調用則值得注意,此時FragA直接從onCreateView調起,也就是說只是重新創建了視圖,而依然使用上次被替換時的Fragment狀態。
OK,說到此時,是否對Fragment的生命周期方法調用在是否addToBackStack時不同有所更加深入的了解了呢?
好吧,最後一個問題。是關於Fragment在FragmentManager管理時,show和hide時的生命周期方法調用。
此時的調用實現方式為:
FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction(); switch (v.getId()) { case R.id.button1: hideAllFrags(fragmentTransaction); if (fragA == null) { fragA = new FragA(); fragmentTransaction.add(R.id.frag_container, fragA, fragNames[0]); fragmentTransaction.addToBackStack(fragNames[0]); } else { fragmentTransaction.show(fragA); } break; case R.id.button2: hideAllFrags(fragmentTransaction); if (fragB == null) { fragB = new FragB(); fragmentTransaction.add(R.id.frag_container, fragB, fragNames[1]); fragmentTransaction.addToBackStack(fragNames[1]); } else { fragmentTransaction.show(fragB); } break; default: break; } fragmentTransaction.commit();
細心的話可以發現,在展示Fragment時,我們使用了方法add而非上面用的replace。而且直接addToBackStack。其實這也可以理解,你想,FragmentManager在show或者hide時,肯定是已經存在的,或者如果沒有的話,需要添加進來Fragment。這便是在show和hide時,需要注意的地方,即使用add和addToBackStack方法。
在點擊按鈕A時,FragA的調用順序為:
01-15 16:57:20.390 9225-9225/com.yeepay.fraglifecircletest I/hideAllFrags﹕ hideAllFrags 01-15 16:57:20.390 9225-9225/com.yeepay.fraglifecircletest I/FragA﹕ onAttach 01-15 16:57:20.390 9225-9225/com.yeepay.fraglifecircletest I/FragA﹕ onCreate 01-15 16:57:20.390 9225-9225/com.yeepay.fraglifecircletest I/FragA﹕ onCreateView 01-15 16:57:20.390 9225-9225/com.yeepay.fraglifecircletest I/FragA﹕ onViewCreated 01-15 16:57:20.390 9225-9225/com.yeepay.fraglifecircletest I/FragA﹕ onActivityCreated 01-15 16:57:20.390 9225-9225/com.yeepay.fraglifecircletest I/FragA﹕ onStart 01-15 16:57:20.390 9225-9225/com.yeepay.fraglifecircletest I/FragA﹕ onResume
可以看出沒有什麼不同於以上所言的部分。
然後,點擊按鈕B時,FragA和FragB的調用順序為:
01-15 16:57:23.360 9225-9225/com.yeepay.fraglifecircletest I/hideAllFrags﹕ hideAllFrags 01-15 16:57:23.360 9225-9225/com.yeepay.fraglifecircletest I/FragB﹕ onAttach 01-15 16:57:23.360 9225-9225/com.yeepay.fraglifecircletest I/FragB﹕ onCreate 01-15 16:57:23.360 9225-9225/com.yeepay.fraglifecircletest I/FragB﹕ onCreateView 01-15 16:57:23.370 9225-9225/com.yeepay.fraglifecircletest I/FragB﹕ onViewCreated 01-15 16:57:23.370 9225-9225/com.yeepay.fraglifecircletest I/FragB﹕ onActivityCreated 01-15 16:57:23.370 9225-9225/com.yeepay.fraglifecircletest I/FragB﹕ onStart 01-15 16:57:23.370 9225-9225/com.yeepay.fraglifecircletest I/FragB﹕ onResume
可以看出,FragA並沒有調用生命周期方法,這說明是展示FragB時,FragA的生命周期並沒有發生變化。而FragB的生命周期與初次點擊按鈕A時FragA的生命周期方法相同。
然後再繼續點擊按鈕A和B,此時打印出來的log為:
1 01-15 16:57:25.220 9225-9225/com.yeepay.fraglifecircletest I/hideAllFrags﹕ hideAllFrags 2 01-15 16:57:44.990 9225-9225/com.yeepay.fraglifecircletest I/hideAllFrags﹕ hideAllFrags 3 01-15 16:57:47.350 9225-9225/com.yeepay.fraglifecircletest I/hideAllFrags﹕ hideAllFrags 4 01-15 16:57:48.020 9225-9225/com.yeepay.fraglifecircletest I/hideAllFrags﹕ hideAllFrags
這說明在FragA和FragB添加進BackStack之後無論如何地show或者hide,它們的生命周期不再發生變化。
而當屏幕上鎖或變暗,然後再解鎖或者變亮時,FragA和FragB的生命周期方法調用順序為:
when screen is locked: ########################################################################################### 01-15 16:58:36.840 9225-9225/com.yeepay.fraglifecircletest I/FragA﹕ onPause 01-15 16:58:36.840 9225-9225/com.yeepay.fraglifecircletest I/FragB﹕ onPause 01-15 16:58:36.870 9225-9225/com.yeepay.fraglifecircletest I/FragA﹕ onStop 01-15 16:58:36.880 9225-9225/com.yeepay.fraglifecircletest I/FragB﹕ onStop when screen is unlocked: ########################################################################################## 01-15 17:05:01.850 9225-9225/com.yeepay.fraglifecircletest I/FragA﹕ onStart 01-15 17:05:01.850 9225-9225/com.yeepay.fraglifecircletest I/FragB﹕ onStart 01-15 17:05:01.870 9225-9225/com.yeepay.fraglifecircletest I/FragA﹕ onResume 01-15 17:05:01.870 9225-9225/com.yeepay.fraglifecircletest I/FragB﹕ onResume
可以看得出來,兩個Fragment都調用了onPause, onStop, onStart, onResume。而且FragA的調用要在FragB之前,這說明跟他們添加進BackStack的順序有關。
以上就是我對Fragment在被FragmentManager管理時,其生命周期方法調用順序的探究,大家覺得如果有什麼地方不嚴謹或者不准確的地方,歡迎在留言處告知。示例工程下載地址為:FragLifeCircleTest
Web開發中Chrome、IE、firefox等浏覽器都自帶提供了插件幫助開發者跟蹤http數據,在手機客戶端怎麼實現http數據抓包 呢?Fiddler可以實現
在Android開發中,我們經常會需要在Android界面上彈出一些對話框,比如詢問用戶或者讓用戶選擇。這些功能我們叫它Android Dialog對話框,在我們
在Android開發中,程序Crash分三種情況:未捕獲的異常、ANR(Application Not Responding)和閃退(NDK引發錯誤)。其中未捕獲
由於目前在做的一款app需要適配手機和平板,所以我在研究怎麼構建可適應所有屏幕尺寸的布局方法。 在web的自適應布局上我有很多經驗,比如使用網格流,CSS3中的