前言
學(xué)習(xí)Java和Android將近一年的時(shí)間了,期間的成果應(yīng)該就是獨(dú)立完成了一個(gè)Android客戶端,并且保證了其在主線版本的穩(wěn)定性。期間遇到了很多坑,也跟著師兄學(xué)到了很多Android知識(shí)。但是人總是要擁抱變化,不能讓自己太安逸,雖然有不舍,但是我已經(jīng)證明了自己的學(xué)習(xí)能力,下一步就是開(kāi)始做Rom Porting了。這里總結(jié)一下之前項(xiàng)目中用到最多的Fragment。
Fragment簡(jiǎn)介
Fragment可以理解成Activity中用戶界面的一個(gè)行為或者一部分,它必須被嵌套在Activity中。但是一個(gè)Fragment有它自己獨(dú)立的xml布局文件,并且具有良好的封裝性,因此特殊情況下Fragment可以很容易用Activity來(lái)進(jìn)行替換。
創(chuàng)建Fragment
創(chuàng)建一個(gè)Fragment和創(chuàng)建Activity類似,需要實(shí)現(xiàn)XML布局文件和Java Class。
XML布局文件和其他布局文件都一樣,例如如下所示的布局文件(fragment_layout.xml):
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" > <TextView android:id="@+id/textView" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@string/testview" /> <Button android:id="@+id/button" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="@string/button" /> </LinearLayout>
Java代碼中,一般情況下可以根據(jù)需要實(shí)現(xiàn)Fragment以下幾個(gè)生命周期方法:
1. onAttach():當(dāng)Fragment依附于activity時(shí)被調(diào)用,可以在該方法中獲取activity句柄,從而實(shí)現(xiàn)Fragment和activity之間的通信。
2. onCreate():對(duì)Fragment做初始化。
3. onCreateView():在第一次為Fragment繪制用戶界面時(shí)系統(tǒng)會(huì)調(diào)用此方法。
4. onActivityCreated():在宿主Activity onCreate函數(shù)執(zhí)行完成之后被調(diào)用,可以在這個(gè)方法里進(jìn)行Fragment自己的widget實(shí)例化和業(yè)務(wù)邏輯處理。
5. onDestoryView():當(dāng)Fragment開(kāi)始被銷毀時(shí)調(diào)用。
6. onStart():當(dāng)Fragment可見(jiàn)時(shí)被調(diào)用。
還有許多其他用以操縱Fragment生命周期中各個(gè)階段的回調(diào)函數(shù),大家可自行Google學(xué)習(xí)。
Fragment生命周期
每一個(gè)Fragment都有自己的一套生命周期回調(diào)方法和處理自己的用戶輸入事件。對(duì)應(yīng)的生命周期如下圖所示:
在Activity中加入Fragment
首先,需要確保Acitivity支持Fragment,因此Activity通常需要繼承自FragmentActivity。在Activity中添加Fragment通常有兩種方法:靜態(tài)的和動(dòng)態(tài)的。
靜態(tài)方法
直接在Activity的XML布局文件中加入Fragment,如下所示:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:baselineAligned="false" android:orientation="horizontal" > <fragment android:id="@+id/first" android:name="com.example.FristFragment" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="1" /> <fragment android:id="@+id/second" android:name="com.example.SecondFragment" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="1" /> </LinearLayout>
<fragment>中的android:name 屬性指定了布局中實(shí)例化的fragment類
當(dāng)系統(tǒng)創(chuàng)建Activity布局時(shí),它實(shí)例化了布局文件中指定的每一個(gè)Fragment,并且為它們調(diào)用onCreateView()函數(shù),以獲取每一個(gè)fragment的布局。系統(tǒng)直接在<fragment>元素位置插入fragment返回的view
注意:每個(gè)fragment都需要一個(gè)唯一的標(biāo)識(shí),如果重啟activity,系統(tǒng)可用來(lái)恢復(fù)fragment(并且用來(lái)捕捉fragment的事務(wù)處理,例如移除)。為了Fragment提供ID有三種方法:
動(dòng)態(tài)方法
使用FragmentTranscation。可以使用FragmentTranscation的API來(lái)對(duì)Activity的Fragment進(jìn)行操作(例如添加,移除,或者替換Fragment)。參考代碼如下:
FragmentManager fragmentManager = getFragmentManager() FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction(); ExampleFragment fragment = new ExampleFragment(); fragmentTransaction.add(R.id.fragment_container, fragment); fragmentTransaction.commit();
傳入add()函數(shù)的第一個(gè)參數(shù)是Fragment被放置的ViewGroup,它由資源ID(resource ID)指定,第二個(gè)參數(shù)就是要添加的fragment。一旦通過(guò)FragmentTranscation做了更改,都應(yīng)當(dāng)使用commit()視變化生效。
Fragments通信
Fragments之間不應(yīng)該直接進(jìn)行通信,它們之間的交互應(yīng)該通過(guò)宿主Activity進(jìn)行。有三種Fragment和Acitivity交互的方法:
1. Activity創(chuàng)建帶參數(shù)的Fragment。
2. Activity中保持了Fragment的對(duì)象句柄,可通過(guò)句柄直接調(diào)用該Fragment的public方法。
3. Fragment可在onAttach函數(shù)中獲取定義的listener句柄。
創(chuàng)建帶參數(shù)的Fragment
在某些特定的情況下,F(xiàn)ragment可能需要特定的參數(shù)來(lái)進(jìn)行初始化。由于Fragment必須只有一個(gè)無(wú)參構(gòu)造函數(shù),因此可以考慮使用靜態(tài)的newInstance方法來(lái)創(chuàng)建帶參數(shù)的Fragment。示例代碼如下:
import android.os.Bundle; import android.support.v4.app.Fragment; public class TestFragment extends Fragment { public static TestFragment newInstance(int num, String title) { TestFragment fragment = new TestFragment(); Bundle args = new Bundle(); args.putInt("num", num); args.putString("title", title); fragment.setArguments(args); return fragment; } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); int num = getArguments().getInt("num", 0); String title = getArguments().getString("title", ""); } }
你可以在Activity里,簡(jiǎn)單的加載一個(gè)帶參數(shù)的Fragment:
FragmentTransaction ft = getSupportFragmentManager().beginTransaction(); TestFragment fragment = TestFragment.newInstance(5, "fragment title"); ft.replace(R.id.placeholder, fragment); ft.commit();
調(diào)用Fragment的方法
因?yàn)锳ctivity可以獲取嵌入的Fragment的句柄,因此可以直接通過(guò)Fragment句柄調(diào)用該方法。
public class TestFragment extends Fragment { public void doSomething(String param) { // do something in fragment } }
在Activity中,可以直接通過(guò)Fragment的對(duì)象句柄調(diào)用該方法:
public class MainActivity extends FragmentActivity { @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); TestFragment testFragment = new TestFragment(); testFragment.doSomething("some param"); } }
Fragment Listener
如果Fragment需要共享事件給Activity,則需要利用這個(gè)方法。Fragment中定義一個(gè)接口,并且由Activity來(lái)實(shí)現(xiàn)這個(gè)接口。在onAttach()方法中將實(shí)現(xiàn)了這個(gè)接口的Activity獲得到。
在Fragment中定義接口代碼如下:
import android.support.v4.app.Fragment; public class MyListFragment extends Fragment { // ... // Define the listener of the interface type // listener is the activity itself private OnItemSelectedListener listener; // Define the events that the fragment will use to communicate public interface OnItemSelectedListener { public void onRssItemSelected(String link); } // Store the listener (activity) that will have events fired once the fragment is attached @Override public void onAttach(Activity activity) { super.onAttach(activity); if (activity instanceof OnItemSelectedListener) { listener = (OnItemSelectedListener) activity; } else { throw new ClassCastException(activity.toString() + " must implement MyListFragment.OnItemSelectedListener"); } } // Now we can fire the event when the user selects something in the fragment public void onSomeClick(View v) { listener.onRssItemSelected("some link"); } }
在Activity中實(shí)現(xiàn)這個(gè)接口:
import android.support.v4.app.FragmentActivity; public class RssfeedActivity extends FragmentActivity implements MyListFragment.OnItemSelectedListener { DetailFragment fragment; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_rssfeed); fragment = (DetailFragment) getSupportFragmentManager() .findFragmentById(R.id.detailFragment); } // Now we can define the action to take in the activity when the fragment event fires @Override public void onRssItemSelected(String link) { if (fragment != null && fragment.isInLayout()) { fragment.setText(link); } } }
新聞熱點(diǎn)
疑難解答
圖片精選