編輯:關於Android編程
在一個Activity中, Fragment代表UI的一個部分或者一個行為。一個Activity可以結合多個Fragment對象,也可以在多個activity中使用相同Fragment字節碼對應的不同對象。一個Fragment對象必須被嵌入在一個主Activity對象中,該Fragment的生命周期與主Activity息息相關。比如,當主Activity處於paused狀態,其對應的所有Fragment對象均處於paused狀態,只有當主Activity處於resumed狀態時,Fragment才能處於自由控制狀態。
為了創建一個Fragment,應該去繼承Fragment或者其子類,覆寫相應的方法。比如onCreate(),OnCreateView(),onPause()等等
實例化一個Fragment對象,除了可以new外,還可以使用Fragment的靜態函數Fragment.instantiate(mContext, "class 完整路徑", info.args);,利用反射實現,但是性能較低
(1).添加UI界面
為該Fragment展現一個布局,必須去實現onCreateView()回掉方法。
[java] view plaincopy
注意:當該Fragment繼承了ListFragment時,不需要覆寫onCreateView()方法,因為默認返回一個ListView對象
- public View onCreateView(LayoutInflater inflater, ViewGroup container,
- Bundle savedInstanceState) {
- View view = inflater.inflate(R.layout.list, null);
- return view;
- }
(2).添加Fragment到Activity
1).通過layout布局文件
android:name屬性應該為Fragment對應類的完整路徑。
[html] view plaincopy
android:id="@+id/f" - android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:name="com.example.a29fragment.MyFragment"/> 在兼容低版本時,如果使用靜態注冊,而MyFragment是使用了兼容support.v4.app.Fragment,就不能使用Activity了,只能使用FragmentActivity
fragment靜態在xml文件配置,該fragment不能被移除,不可動態被編輯。
[html] view plaincopy
android:layout_width="match_parent" - android:layout_height="match_parent"
- android:orientation="horizontal" >
android:id="@+id/list" - android:name="com.example.news.ArticleListFragment"
- android:layout_width="0dp"
- android:layout_height="match_parent"
- android:layout_weight="1" />
android:id="@+id/viewer" - android:name="com.example.news.ArticleReaderFragment"
- android:layout_width="0dp"
- android:layout_height="match_parent"
- android:layout_weight="2" />
2).通過Java代碼
當Activity運行時,可以自由的在該activity上添加fragment對象,但應該指定一個ViewGroup容器,可以FragmentTransaction完成fragment的添加移除或者替換。
[java] view plaincopy
- manager = getFragmentManager();
- if(manager.findFragmentByTag("right") == null){
- manager.beginTransaction().replace(R.id.right, new RightFrag(), "right").commit();
- }
(3).fragment唯一標示符
每個fragment需要定義一個唯一的標識符,如果activity被銷毀又重新啟動,系統能夠恢復該fragment的狀態。如果想重新恢復,需滿足下面有3種方式之一:
1).定義ID
在布局文件中,定義android:id屬性
[html] view plaincopy
android:id="@+id/list" - android:name="com.example.news.ArticleListFragment"
- android:layout_width="0dp"
- android:layout_height="match_parent"
- android:layout_weight="1" />
2).指明tag
android:tag 指明 或者 一個fragment對象add()或者replace()時指定tag
[html] view plaincopy
android:id="@+id/list" - android:tag="first"
- android:name="com.example.news.ArticleListFragment"
- android:layout_width="0dp"
- android:layout_height="match_parent"
- android:layout_weight="1" />
或者
[java] view plaincopy
- manager.beginTransaction()
- .replace(R.id.right, new RightFrag(), "right")//在事務中指明該fragment的tag
- .commit();
3).viewgroup ID
如果該fragment均沒有id和tag,系統將使用container view布局的id
通過getFragmentManager()方法,可以得到FragmentManager對象,主要完成下面的功能
[java] view plaincopy
- FragmentManager manager = getFragmentManager();
(1).得到已經存在Fragment對象
如果該fragment在布局文件中指定了id,通過findFragmentById()得到對象,或者指定了tag可以通過findFragmentByTag()得到對象
[java] view plaincopy
- Fragment fragment = getFragmentManager().findFragmentByTag("right");
- //or
- Fragment fragment = getFragmentManager().findFragmentById(id);
(2).注冊OnBackStackChangedListener監聽器
可以用來監聽該任務對應的返回棧信息,當該返回棧狀態發生改變時,執行對應的onBackStackChanged() 方法
[java] view plaincopy
- manager.addOnBackStackChangedListener(new FragmentManager.OnBackStackChangedListener() {
- @Override
- public void onBackStackChanged() {
- Toast.makeText(MainActivity.this, "返回堆狀態發生改變", 1).show();
- }
- });
(3).彈出返回棧
模擬用戶點擊返回鍵,將指定的fragment從返回棧中彈出,該操作為異步的。前提是該fragment對象使用.beginTransaction().addToBackStack("right")添加了進返回棧
[java] view plaincopy
- manager.popBackStack(); //Pop the top state off the back stack
(4).FragmentTransaction事務
事務主要包含一些操作的集合,比如增加移除替換,動畫設置等等
[html] view plaincopy
- /*
- * 通過manager開啟一個事務,該事務包含一些操作的集合,通事務可以 add(), remove(), replace()
- * 完成對Fragment的操作,並使用commit()提交
- */
- FragmentTransaction transaction = manager.beginTransaction();
- transaction.replace(R.id.right, new RightFrag(), "right");
- transaction.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);//設置動畫
- transaction.addToBackStack("right"); // 將該fragment加入返回堆
- // 提交事務
- transaction.commit();
(5).Fragment狀態管理
[java] view plaincopy
- /*
- * 管理Fragment的狀態
- * 如果在一個主activityViewGroup中添加一個fragment,
- * 如果手機屏幕旋轉了,當前activity被銷毀重建,fragment也被activityManager創建
- * 故在onCreate中,需要判斷一下
- */
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- manager = getFragmentManager();
- if (manager.findFragmentByTag("right") == null) {
- // if(savedInstanceState == null)也可判斷該fragment是否已經加載
- manager.beginTransaction()
- .replace(R.id.right, new RightFrag(), "right")
- .setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN)// 設置動畫
- .addToBackStack("right") // 將該fragment加入返回堆
- // 提交事務
- .commit();
- }
- }
可參閱CursorLoader的兼容~~,特別注意:如果使用靜態注冊,在布局文件配置標簽時,指定了name的class由於兼容support.v4.app.Fragment,加載布局文件的Class就不能繼承Activity了,只能繼承FragmentActivity
(1).取得對象
[java] view plaincopy
- /*
- * 點擊該Fragment的button按鈕,將該button的text設置為另一個fragment中Edittext的文本值
- */
- public View onCreateView(LayoutInflater inflater, ViewGroup container,
- Bundle savedInstanceState) {
- View view = inflater.inflate(R.layout.list, null);
- final Button button = (Button) view.findViewById(R.id.confirm);
- button.setOnClickListener(new View.OnClickListener() {
- @Override
- public void onClick(View v) {
- //通過FragmentManager找到另一個fragment中的edittext對象,並取得text內容
- EditText editText = (EditText)(getFragmentManager().findFragmentByTag("left").getView().findViewById(R.id.name));
- button.setText(editText.getText().toString());
- }
- });
- return view;
- }
(2).通回掉函數
[java] view plaincopy
- public class MainActivity extends Activity {
- private FragmentManager manager;
- private Button button;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- button.setOnClickListener(new View.OnClickListener() {
- @Override
- public void onClick(View v) {
- RightFragment rightFrag = (RightFragment) (getFragmentManager().findFragmentByTag("right"));
- /*
- * 通過set方法,向其傳遞一個實例化對象,由於rightFrag.set()方法內部執行RightFragment.CallBack.get()方法,完成了參數的傳遞
- */
- rightFrag.set(new RightFragment.CallBack() {
- @Override
- public void get(String str) {
- button.setText(str);
- }
- });
- }
- });
- }
- } [java] view plaincopy
- public class RightFragment extends ListFragment {
- private LoaderManager manager;
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- manager = getLoaderManager();
- }
- /*
- * 點擊該Fragment的button按鈕,將該button的text設置為另一個fragment中Edittext的文本值
- */
- public View onCreateView(LayoutInflater inflater, ViewGroup container,
- Bundle savedInstanceState) {
- View view = inflater.inflate(R.layout.list, null);
- return view;
- }
- /**
- * 通過調用該方法,接收一個回掉函數對象,callBack.get(str);
- * @param callBack
- */
- public void set(CallBack callBack) {
- EditText editText = (EditText) getView().findViewById(R.id.name);
- callBack.get(editText.getText().toString());
- }
- /*
- * 回掉接口
- */
- interface CallBack {
- public void get(String str);
- }
- }
(1).生命周期路線圖
生命狀態
周期過程
foreground lifetime
onResume(A)
onResume(F)
onPause(F)
onPause(A)
visible lifetime
onCreateView(F)
onActivityCreated(F)
onStart(A)
onStart(F)
onResume(A)
onResume(F)
onPause(F)
onPause(A)
onStop(F)
onStop(A)
onDestroyView(F)
entire lifetime
完整的生命周期
(2).生命周期回掉函數概述
方法 描述 onAttach(Activity)
當前Fragment與Activity關聯,調用!
onCreate()完成fragment的初始化創建
onCreateView()創建並返回與當前fragment相關聯的層次視圖view
onActivityCreated()主activity的onCreate()執行完後,該方法才執行
onStart()fragment可見,當主activity處於started狀態後執行
onResume()fragment能與用戶交互,當主activity處於resumed狀態後執行
onPause()fragment不在與用戶交互,可能在主activity將要處於paused前執行,可能該fragment被修改
onStop()fragment不在可見,可能在主activity將要處於stopped前執行,可能該fragment被修改
onDestroyView()允許該fragment清理視圖相關資源
onDestroy()清理掉視圖state信息
onDetach()該fragment不在於activity關聯
本博文為子墨原創,轉載請注明出處! http://blog.csdn.net/zimo2013/article/details/12239349
經驗解決Fragment 被 Replace後仍舊可見的問題
網上問的問題,大多會提到替換了Fragment而發現之前被替換的仍舊顯示在那裡。我個人使用android 2.3 +support 開發包,在2.3系統上也出現類似問題。搜了下網上的問題,好像都沒有找到解決方法。之後自己摸索。最後發現其實,對於Fragment的替換 JAVA代碼基本上沒啥,網上都是正確的,比如:
- FragmentManager fragmentManager = getSupportFragmentManager();
- FragmentTransaction transaction = fragmentManager.beginTransaction();
- OrderFragment orderFragment = new OrderFragment();
- Bundle args = new Bundle();
- args.putInt("card_id", LoginHelper.currentCard.getId());
- args.putBoolean("create_order", true);
- orderFragment.setArguments(args);
- transaction.replace(R.id.layout_shopping1, orderFragment);
- //transaction.addToBackStack(null);
- transaction.commit(); 復制代碼 但多數人並沒有意識到,貼出 XML布局文件的重要性:
正確的做法是必須使用FrameLayout作為Fragment被替換的布局容器
例如:
- <frameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/layout_shopping1"
- android:layout_width="match_parent"
- android:layout_height="wrap_content"
- android:background="@color/black" >
- </frameLayout> 復制代碼 不能夠使用比如線性布局LinearLayout 等,否則就會發生看得見的問題。
奉獻點經驗,期望可以幫助到遇到類似問題的開發者。
前言 本文實現的效果:文本框輸入為空時顯示輸入的圖標;不為空時顯示清空的圖標,此時點擊清空圖標能清空文本框內輸入文字。正文 一、實現效果 二、實現代碼 綁定事件 復
1.ServicesService 是一個可以在後台執行長時間運行操作而不使用用戶界面的應用組件。服務可由其他應用組件啟動,而且即使用戶切換到其他應用,服務仍將在後台繼續
Android 網絡直播彈幕  
最近項目中需要客戶端往服務器傳輸圖片,並且還需要附帶一些普通參數,研究了幾天,把結果記錄下。 首先客戶端可服務端進行通信一般都是有http請求來發送和接收數據,這裡an