編輯:關於Android編程
回調函數
就像activities一樣,fragments也有它們自己的生命周期。理解fragments的生命周期,可以使你在它們被銷毀的時候保存它們的實例,這樣在它們重新被創建的時候,就能恢復它們之前的狀態。
流程:
onAttach()
作用:fragment已經關聯到activity,
這個是 回調函數
@Override public void onAttach(Activity activity) { super.onAttach(activity); Log.i("onAttach_Fragment"); }
這個時候 activity已經傳進來了, 獲得activity的傳遞的值 就可以進行 與activity的通信裡, 當然也可以使用getActivity(),前提是這個fragment已經和宿主的activity關聯,並且沒有脫離,他只調用一次。
onCreate()
系統創建fragment的時候回調他,在他裡面實例化一些變量
這些個變量主要是:當你 暫停 停止的時候 你想保持的數據
如果我們要為fragment啟動一個後台線程,可以考慮將代碼放於此處。
參數是:Bundle savedInstance, 用於保存 Fragment 參數, Fragement 也可以 重寫 onSaveInstanceState(BundleoutState) 方法, 保存Fragement狀態;
可以用於 文件保護
他只調用一次。
onCreateView()
第一次使用的時候 fragment會在這上面畫一個layout出來, 為了可以畫控件 要返回一個 布局的view,也可以返回null。
當系統用到fragment的時候 fragment就要返回他的view,越快越好 ,所以盡量在這裡不要做耗時操作,比如從數據庫加載大量數據顯示listview,
當然線程還是可以的。
給當前的fragment繪制ui布局,可以使用線程更新UI,說白了就是加載fragment的布局的。
這裡一般都先判斷是否為null。
if(text==null){ Bundle args=getArguments(); text=args.getString("text"); } if (view == null) { view = inflater.inflate(R.layout.hello, null); }
這樣進行各判斷省得每次都要加載,減少資源消耗
onActivityCreated()
當Activity中的onCreate方法執行完後調用。
注意了:
從這句官方的話可以看出:當執行onActivityCreated()的時候 activity的
onCreate才剛完成。
所以在onActivityCreated()調用之前 activity的onCreate可能還沒有完成,
所以不能再onCreateView()中進行 與activity有交互的UI操作,UI交互操作可以在onActivityCreated()裡面進行。
所以呢:這個方法主要是初始化那些你需要你的父Activity或者Fragment的UI已經被完
整初始化才能初始化的元素。
如果在onCreateView裡面初始化空間 會慢很多,比如listview等。
onStart()
和activity一致,啟動Fragement 啟動時回調,,此時Fragement可見。
onResume()
和activity一致 在activity中運行是可見的。激活, Fragement 進入前台, 可獲取焦點時激活。
onPause()
和activity一致 其他的activity獲得焦點,這個仍然可見
第一次調用的時候,指的是 用戶 離開這個fragment(並不是被銷毀)
通常用於 用戶的提交(可能用戶離開後不會回來了)
onStop()
和activity一致, fragment不可見的, 可能情況:activity被stopped了OR fragment被移除但被,加入到回退棧中,一個stopped的fragment仍然是活著的如果長時間不用也會被移除。
onDestroyView()
Fragment中的布局被移除時調用。
表示fragemnt銷毀相關聯的UI布局, 清除所有跟視圖相關的資源。
以前以為這裡沒什麼用處其實 大有文章可做,
相信大家都用過ViewPager+Fragment,由於ViewPager的緩存機制,每次都會加載3
頁。
例如:有四個 fragment 當滑動到第四頁的時候 第一頁執行onDestroyView(),但沒有
執行onDestroy。他依然和activity關聯。當在滑動到第一頁的時候又執行了
onCreateView()。 生命周期可以自己試一下。
那麼問題來了。會出現重復加載view的局面,所以這麼做(下面是先人的代碼)
@Override public void onDestroyView() { Log.i("onDestroyView_Fragment"); if(view!=null){ ((ViewGroup)view.getParent()).removeView(view); } super.onDestroyView(); }
onDestroy()
銷毀fragment對象, 跟activity類似了。
onDetach()
Fragment和Activity解除關聯的時候調用。 脫離activity。
可見fragment的銷毀還是很優雅地,一個一個的來。
下面貼一下 activity和fragment同時運行時候的 生命周期
開始啟動:
05-07 05:55:08.553: I/Log(1990): oncreate 05-07 05:55:08.553: I/Log(1990): onAttach_Fragment 05-07 05:55:08.553: I/Log(1990): onCreate_Fragment 05-07 05:55:08.553: I/Log(1990): onCreateView_Fragment 05-07 05:55:08.553: I/Log(1990): onActivityCreated_Fragment 05-07 05:55:08.553: I/Log(1990): onStart 05-07 05:55:08.553: I/Log(1990): onStart_Fragment 05-07 05:55:08.553: I/Log(1990): onResume 05-07 05:55:08.553: I/Log(1990): onResume_Fragment
按下home按鍵
05-07 05:55:28.725: I/Log(1990): onPause_Fragment 05-07 05:55:28.725: I/Log(1990): onPause 05-07 05:55:29.221: I/Log(1990): onStop_Fragment 05-07 05:55:29.221: I/Log(1990): onStop
再回到界面
05-07 05:55:49.441: I/Log(1990): onRestart 05-07 05:55:49.441: I/Log(1990): onStart 05-07 05:55:49.441: I/Log(1990): onStart_Fragment 05-07 05:55:49.441: I/Log(1990): onResume 05-07 05:55:49.441: I/Log(1990): onResume_Fragment
銷毀activity
05-07 05:59:02.293: I/Log(1990): onPause_Fragment 05-07 05:59:02.293: I/Log(1990): onPause 05-07 05:59:02.757: I/Log(1990): onStop_Fragment 05-07 05:59:02.757: I/Log(1990): onStop 05-07 05:59:02.757: I/Log(1990): onDestroyView_Fragment 05-07 05:59:02.757: I/Log(1990): onDestroy_Fragment 05-07 05:59:02.757: I/Log(1990): onDetach_Fragment 05-07 05:59:02.757: I/Log(1990): onDestroy
可以看出 當現實fragment的時候都先執行activity方法,當銷毀的時候都是現執行 fragment的方法,這樣更好理解fragment是嵌套在activity中
下面一個綜合性的例子測試了fragments的不同狀態:
1.創建一個Fragment的子類:Fragment1.java。
package net.horsttnann.Fragments; 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; public class Fragment1 extends Fragment { @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { Log.d("Fragment 1", "onCreateView"); // ---Inflate the layout for this fragment--- return inflater.inflate(R.layout.fragment1, container, false); } @Override public void onAttach(Activity activity) { super.onAttach(activity); Log.d("Fragment 1", "onAttach"); } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Log.d("Fragment 1", "onCreate"); } @Override public void onActivityCreated(Bundle savedInstanceState) { super.onActivityCreated(savedInstanceState); Log.d("Fragment 1", "onActivityCreated"); } @Override public void onStart() { super.onStart(); Log.d("Fragment 1", "onStart"); } @Override public void onResume() { super.onResume(); Log.d("Fragment 1", "onResume"); } @Override public void onPause() { super.onPause(); Log.d("Fragment 1", "onPause"); } @Override public void onStop() { super.onStop(); Log.d("Fragment 1", "onStop"); } @Override public void onDestroyView() { super.onDestroyView(); Log.d("Fragment 1", "onDestroyView"); } @Override public void onDestroy() { super.onDestroy(); Log.d("Fragment 1", "onDestroy"); } @Override public void onDetach() { super.onDetach(); Log.d("Fragment 1", "onDetach"); } }
2.按Ctrl+F11,將模擬器轉換成“橫屏模式”。
3.按F11調試。
4.當應用被加載的時候,LogCat窗口中有如下顯示。
03-27 12:23:32.255: D/Fragment 1(704): onAttach 03-27 12:23:32.255: D/Fragment 1(704): onCreate 03-27 12:23:32.255: D/Fragment 1(704): onCreateView 03-27 12:23:32.274: D/Fragment 1(704): onActivityCreated 03-27 12:23:32.274: D/Fragment 1(704): onStart 03-27 12:23:32.286: D/Fragment 1(704): onResume
5.按Home鍵,LogCat窗口中有如下顯示。
03-27 12:25:23.174: D/Fragment 1(704): onPause 03-27 12:25:25.174: D/Fragment 1(704): onStop
6.按Home鍵不放,重新進入應用,LogCat窗口中有如下顯示。
03-27 12:26:21.505: D/Fragment 1(704): onStart 03-27 12:26:21.505: D/Fragment 1(704): onResume
7.按返回鍵,LogCat窗口中有如下顯示。
03-27 12:27:54.384: D/Fragment 1(704): onPause 03-27 12:27:55.324: D/Fragment 1(704): onStop 03-27 12:27:55.324: D/Fragment 1(704): onDestroyView 03-27 12:27:55.324: D/Fragment 1(704): onDestroy 03-27 12:27:55.324: D/Fragment 1(704): onDetach
解析:
1.當一個fragment被創建的時候,它會經歷以下狀態.。
onAttach()
onCreate()
onCreateView()
onActivityCreated()
2.當這個fragment對用戶可見的時候,它會經歷以下狀態。
onStart()
onResume()
3.當這個fragment進入“後台模式”的時候,它會經歷以下狀態。
onPause()
onStop()
4.當這個fragment被銷毀了(或者持有它的activity被銷毀了),它會經歷以下狀態。
onPause()
onStop()
onDestroyView()
onDetach()
5.就像activitie一樣,在以下的狀態中,可以使用Bundle對象保存一個fragment的對象。
onCreate()
onCreateView()
onActivityCreated()
6.fragments的大部分狀態都和activitie很相似,但fragment有一些新的狀態。
onAttached() —— 當fragment和activity關聯之後,調用這個方法。
onCreateView() —— 創建fragment中的視圖的時候,調用這個方法。
onActivityCreated() —— 當activity的onCreate()方法被返回之後,調用這個方法。
onDestroyView() —— 當fragment中的視圖被移除的時候,調用這個方法。
onDetach() —— 當fragment和activity分離的時候,調用這個方法。
前言:為什麼要了解系統Activity,Service,BroadCastReceiver,ContentProvider的啟動流程,這是一個對於即將理解插件中的四大組件
第一:了解三個類Canvas,在英語中,這個單詞的意思是帆布。在Android中,則把Canvas當做畫布,只要我們借助設置好的畫筆(Paint類)就可以在畫布上繪制我們
三種得到LinearInflater的方法a. LayoutInflater inflater = getLayoutInflater();b. LayoutInflat
本部分代碼在《Android應用開發揭秘》中提到,但是在eclipse環境下調試時出現異常,幾番糾結,代碼終於可以播放器音樂、並成功移植到手機上...... p