學習筆記| AS入門(六) 碎片Fragment

安卓自3.0開始引入Fragment的概念,主要是爲了能在不同分辯率屏幕上進行更爲動態和靈活的UI設計,讓程序更加合理和充分利用大屏幕空間。本篇將學習Fragment以下幾個知識點:

  • Fragment概要
  • Fragment生命週期
  • 加載Fragment方法
    • 靜態加載
    • 動態加載
  • Fragment與Activity之間通信

1.Fragment概要

學習Fragment的時候可以聯繫之前學習過的Activity,因爲它們有很大相似點:都可包含佈局,有自己的生命週期,Fragment可看似迷你活動。正如Fragment的名字–碎片,它的出現是爲了解決Android碎片化 ,它可作爲Activity界面的組成部分,可在Activity運行中實現動態地加入、移除和交換。一個Activity中可同時出現多個Fragment,一個Fragment也可在多個Activity中使用。活動和碎片像極了夫妻, 雖然緊密聯繫但是又有獨立空間,在一起讓彼此變得更好。

下面這個非常經典的例子更直觀地說明了Fragment作用:

2.Fragment生命週期

先來看官方文檔提供的有關Fragment生命週期的圖片。

是不是能發現Fragment和Activity的生命週期太相似了,現在只需要再介紹幾個Activity中沒講過的新方法:

onAttach():當Fragment和Activity建立關聯時調用
onCreateView():當Fragment創建視圖時調用
onActivityCreated():當與Fragment相關聯的Activity完成onCreate()之後調用
onDestroyView():在Fragment中的佈局被移除時調用
onDetach():當Fragment和Activity解除關聯時調用

在上圖中畫了幾條線,可以看到Fragment週期中的狀態幾乎都是成對出現的,所以不難理解下圖幾種變化下Fragment生命週期方法的調用順序了。

加載到Activity中的Fragment在各種變化下方法的調用順序更值得注意。需要提一句的是,Activity的FragmentManager負責調用隊列中Fragment的生命週期方法,只要Fragment的狀態與Activity的狀態保持了同步,託管Activity的FragmentManager便會繼續調用其他生命週期方法以繼續保持Fragment與Activity的狀態一致。

Fragment生命週期與Activity生命週期的一個關鍵區別就在於,Fragment的生命週期方法是由託管Activity而不是操作系統調用的。Activity中生命週期方法都是protected,而Fragment都是public,也能印證了這一點,因爲Activity需要調用Fragment那些方法並管理它。

3.加載Fragment方法

現在就來學習如何在Activity中加載Fragment。

(1)靜態加載:在託管Activity的layout文件中聲明Fragment

靜態加載Fragment大致過程如圖,分成四步:

下面通過一個簡單的例子感受Fragement靜態加載到Activity的過程。

第一步:新建frag_layout.xml,爲Fragment指定一個佈局,這裏簡單的放一個TextView和一個Button。

第二步:新建一個MyFragment類並繼承Fragment,這裏引用android.app包下的就可以,另一個包下主要用於兼容低版本的安卓系統。然後重寫onCreateView()方法,這個方法裏通過LayoutInflater的inflate()方法將剛剛定義的frag_layout佈局加載進來並得到一個View,再return這個View。

第三步:新建mian.xml,作爲Activity的佈局,使用< fragment>標籤添加碎片,並且一定要有android:name屬性且值爲被加載的Fragment類的包名.類名完整形式。

第四步:在MainActivity中加載main佈局。現在MyFragment就完成了靜態加載到MainActivity中,這時碎片裏的控件自然也是活動的一個部分,可直接在活動中獲取到Button的實例,來註冊點擊事件了。

運行一下看看能不能達到效果:

(2)動態加載:在託管Activity通過代碼動態添加

動態加載的代碼也非常簡單,直接看例子。修改main.xml,將整個< fragment>刪掉。但還保留一個LinerLayout的空間並且還給了Id,爲何這樣做?馬上揭祕。

接下來在MainActivity中添加幾行代碼:

可將整個過程大致分爲三個步驟:

第一步,先用getFragmentManager()方法獲取一個FragmentManager對象,再通過它的beginTransaction()獲取一個FragmentTransaction的實例。

第二步,用beginTransaction.add()方法將MyFragemnt實例添加到main佈局裏LinearLayout裏,終於知道之前鋪墊的Id是怎麼回事了。一定要注意,add()方法裏的第一個參數是容器視圖資源Id,而不是layout。容器視圖資源Id有兩個作用:告知FragmentManager,碎片視圖應該出現在活動視圖的什麼地方;它也是FragmentManager隊列中碎片的唯一標識符。而靜態加載時碎片的唯一標識符正是在活動佈局裏< fragment>下的id。

第三步:調用beginTransaction.commit()提交。另外,如果允許用戶通過按下返回按鍵返回到前一個Fragment狀態,在調用commit()之前先調用addToBackStack(true)方法。

這裏注意到動態加載進來的Fragment裏的控件並不能直接在活動中findViewById得到,那麼如何實現點擊效果呢,學完下一個知識點就有辦法了。

4.Fragment與Activity之間通信

在活動中可以通過調用FragmentManagerfindFragmentById()方法來得到相應碎片的實例,繼而可以調用碎片裏的方法。以上面demo舉例,如果想得到靜態加載碎片的實例,可在MainActivity添加代碼如下:

MyFragment myFragment = (MyFragment)getFragmentManager(). findFragmentById(R.id.fragment);

如果想得到動態加載碎片的實例,代碼如下:

MyFragment myFragment = (MyFragment)fragmentManager(). findFragmentById(R.id.layout);

在碎片中也可以通過調用getActivity()方法來得到和當前碎片相關聯的活動實例,這樣調用活動裏的方法就變得輕而易舉了。比如想在MyFragment得到MainActivity的實例:

MainActivity activity=(MainActivity)getActivity();

於是碎片和活動可以很方便地進行通信了。再想一想碎片和碎片之間如何進行通信?先在一個碎片中可以得到與它相關聯的活動,然後再通過這個活動去獲取另外一個碎片的實例,這樣實現了不同碎片之間的通信了。

現在你有沒有想到解決之前那個問題的辦法呢?可以這樣做,修改MyFragment,代碼如下圖所示:

現在按鈕點擊就又有響應了!其實在實際開發中,如果某一板塊每次用到都需要相同的功能,就完全可以在碎片中實現需求,而不必在活動中重複代碼,這樣可以提高代碼的複用性。

> 下一篇預告:數據存儲篇

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章