在上一节中讲了静态加载,使用静态加载的场景非常局限,因此这里详细讲解动态加载Fragment,这也是开发中常用的使用场景。Fragment是activity的界面中的一部分或一种行为。你可以把多个Fragment们组合到一个activity中来创建一个多面界面并且你可以在多个activity中重用一个Fragment。你可以把Fragment认为模块化的一段activity,它具有自己的生命周期,接收它自己的事件,并可以在activity运行时被添加或删除。
一 生命周期
这里用一张图详细说明Fragment的生命周期。
通常这几个方法比较常用::
onCreate()
The system calls this when creating the fragment. Within your implementation, you should initialize essential components of the fragment that you want to retain when the fragment is paused or stopped, then resumed.
在创建fragment时系统会调用此方法。在实现代码中,你可以初始化想要在fragment中保持的那些必要组件(这里的组件是指除了view之外的东西,比如需要进行界面展示的关键数据),当fragment处于暂停或者停止状态之后可重新启用它们。
onCreateView()
The system calls this when it's time for the fragment to draw its user interface for the first time. To draw a UI for your fragment, you must return a View from this method that is the root of your fragment's layout. You can return null if the fragment does not provide a UI.
在第一次为fragment绘制用户界面时系统会调用此方法。为fragment绘制用户界面,这个函数必须要返回所绘出的fragment的根View。如果fragment没有用户界面可以返回空。
onPause()
The system calls this method as the first indication that the user is leaving the fragment (though it does not always mean the fragment is being destroyed). This is usually where you should commit any changes that should be persisted beyond the current user session (because the user might not come back).
系统回调用该函数作为用户离开fragment的第一个预兆(尽管这并不总意味着fragment被销毁)。在当前用户会话结束之前,通常要在这里提交任何应该持久化的变化(因为用户可能不再返回)。
二 布局文件
activity_main.xml
1. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
2. xmlns:tools="http://schemas.android.com/tools"
3. android:layout_width="match_parent"
4. android:orientation="vertical"
5. android:layout_height="match_parent"
6. tools:context="com.example.dfragmentdemo.MainActivity" >
7.
8. <FrameLayout android:layout_width="match_parent"
9. android:layout_weight="1"
10. android:id="@+id/content"
11. android:layout_height="match_parent"></FrameLayout>
12.
13. <View android:layout_width="match_parent"
14. android:layout_height="1dp"
15. android:background="#000"/>
16. <LinearLayout android:layout_width="match_parent"
17. android:layout_height="50dp"
18. android:orientation="horizontal"
19. android:gravity="center_vertical"
20. android:layout_alignParentBottom="true">
21. <ImageView android:layout_width="40dp"
22. android:layout_height="40dp"
23. android:layout_weight="1"
24. android:id="@+id/tuijian_iv"
25. android:src="@drawable/tuijian1"/>
26. <ImageView android:layout_width="40dp"
27. android:layout_height="40dp"
28. android:layout_weight="1"
29. android:id="@+id/daohang_iv"
30. android:src="@drawable/daohang1"/>
31. <ImageView android:layout_width="40dp"
32. android:layout_height="40dp"
33. android:layout_weight="1"
34. android:id="@+id/faxian_iv"
35. android:src="@drawable/faxian1"/>
36. <ImageView android:layout_width="40dp"
37. android:layout_height="40dp"
38. android:layout_weight="1"
39. android:id="@+id/wode_iv"
40. android:src="@drawable/wode1"/>
41. <ImageView android:layout_width="40dp"
42. android:layout_height="40dp"
43. android:layout_weight="1"
44. android:id="@+id/lixian_iv"
45. android:src="@drawable/lixian1"/>
46. </LinearLayout>
47.
48. </LinearLayout>
tuijian_main.xml
1. <?xml version="1.0" encoding="utf-8"?>
2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
3. android:layout_width="match_parent"
4. android:layout_height="match_parent"
5. android:background="#00ff00" >
6.
7. <TextView android:layout_width="wrap_content"
8. android:layout_height="wrap_content"
9. android:text="推荐"
10. android:textSize="30sp"
11. android:layout_centerInParent="true"/>
12. </RelativeLayout>
还有其他四个Fragment布局文件和tuijian_main.xml类似,这里就不粘贴了。
三 创建Fragment
创建Tuijian.Java,继承import Android.support.v4.app.Fragment,并实现onCreateView(),其他的方法先作为日志输出查看生命周期使用。
1. package com.example.dfragmentdemo;
2.
3. import android.content.Context;
4. import android.os.Bundle;
5. import android.support.annotation.Nullable;
6. import android.support.v4.app.Fragment;
7. import android.util.Log;
8. import android.view.LayoutInflater;
9. import android.view.View;
10. import android.view.ViewGroup;
11.
12. public class Tuijian extends Fragment {
13. @Override
14. @Nullable
15. public View onCreateView(LayoutInflater inflater,
16. @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
17. // TODO Auto-generated method stub
18. "onCreateView");
19. return inflater.inflate(R.layout.tuijian_main, container, false);
20. }
21. private static final String ACTIVITY_TAG="TuijianFragment";
22. @Override
23. public void onAttach(Context context) {
24. // TODO Auto-generated method stub
25. super.onAttach(context);
26. "onAttach");
27. }
28.
29. @Override
30. public void onCreate(@Nullable Bundle savedInstanceState) {
31. // TODO Auto-generated method stub
32. super.onCreate(savedInstanceState);
33. "onCreate");
34. }
35.
36. @Override
37. public void onActivityCreated(@Nullable Bundle savedInstanceState) {
38. // TODO Auto-generated method stub
39. super.onActivityCreated(savedInstanceState);
40. "onActivityCreated");
41. }
42.
43. @Override
44. public void onStart() {
45. // TODO Auto-generated method stub
46. super.onStart();
47. "onStart");
48. }
49.
50. @Override
51. public void onResume() {
52. // TODO Auto-generated method stub
53. super.onResume();
54. "onResume");
55. }
56.
57. @Override
58. public void onPause() {
59. // TODO Auto-generated method stub
60. super.onPause();
61. "onPause");
62. }
63.
64. @Override
65. public void onStop() {
66. // TODO Auto-generated method stub
67. super.onStop();
68. "onStop");
69. }
70.
71. @Override
72. public void onDestroyView() {
73. // TODO Auto-generated method stub
74. super.onDestroyView();
75. "onDestroyView");
76. }
77.
78. @Override
79. public void onDestroy() {
80. // TODO Auto-generated method stub
81. super.onDestroy();
82. "onDestroy");
83. }
84. @Override
85. public void onDetach() {
86. // TODO Auto-generated method stub
87. super.onDetach();
88. "onDetach");
89. }
90. }
其他的四个Fragment也是差不多,这里也不粘贴了。
四 在MainActiviy中实现动态加载Fragment
通过点击导航栏按钮中的button即可加载不同的Fragment
1. package com.example.dfragmentdemo;
2.
3. import android.os.Bundle;
4. import android.support.v4.app.FragmentActivity;
5. import android.support.v4.app.FragmentManager;
6. import android.support.v4.app.FragmentTransaction;
7. import android.view.Menu;
8. import android.view.MenuItem;
9. import android.view.View;
10. import android.view.View.OnClickListener;
11. import android.widget.ImageView;
12.
13. public class MainActivity extends FragmentActivity {
14. private ImageView tuijian_iv;
15. private ImageView daohang_iv;
16. private ImageView faxian_iv;
17. private ImageView wode_iv;
18. private ImageView lixian_iv;
19. FragmentManager fragmentManager;
20. FragmentTransaction fragmentTransaction;
21. private Tuijian tuijian;
22. private Daohang daohang;
23. private Faxian faxian;
24. private Wode wode;
25. private Lixian lixian;
26. @Override
27. protected void onCreate(Bundle savedInstanceState) {
28. super.onCreate(savedInstanceState);
29. setContentView(R.layout.activity_main);
30. tuijian_iv = (ImageView) findViewById(R.id.tuijian_iv);
31. daohang_iv = (ImageView) findViewById(R.id.daohang_iv);
32. faxian_iv = (ImageView) findViewById(R.id.faxian_iv);
33. wode_iv = (ImageView) findViewById(R.id.wode_iv);
34. lixian_iv = (ImageView) findViewById(R.id.lixian_iv);
35.
36. tuijian_iv.setImageResource(R.drawable.tuijian2);
37.
38. new MyOnClickListener());
39. new MyOnClickListener());
40. new MyOnClickListener());
41. new MyOnClickListener());
42. new MyOnClickListener());
43.
44. fragmentManager = getSupportFragmentManager();
45. 0);
46.
47. }
48. private void RemoveFragment(FragmentTransaction fragmentTransaction){
49. if (tuijian!=null) {
50. fragmentTransaction.remove(tuijian);
51. }
52. if (daohang!=null) {
53. fragmentTransaction.remove(daohang);
54. }
55. if (faxian!=null) {
56. fragmentTransaction.remove(faxian);
57. }
58. if (wode!=null) {
59. fragmentTransaction.remove(wode);
60. }
61. if (lixian!=null) {
62. fragmentTransaction.remove(lixian);
63. }
64. }
65.
66. private void HideFragment(FragmentTransaction fragmentTransaction){
67. if (tuijian!=null) {
68. fragmentTransaction.hide(tuijian);
69. }
70. if (daohang!=null) {
71. fragmentTransaction.hide(daohang);
72. }
73. if (faxian!=null) {
74. fragmentTransaction.hide(faxian);
75. }
76. if (wode!=null) {
77. fragmentTransaction.hide(wode);
78. }
79. if (lixian!=null) {
80. fragmentTransaction.hide(lixian);
81. }
82. }
83. private void SelectFragment(int index) {
84. fragmentTransaction = fragmentManager
85. .beginTransaction();
86. //RemoveFragment(fragmentTransaction);
87. HideFragment(fragmentTransaction);
88. switch (index) {
89. case 0:
90. if (tuijian==null) {
91. new Tuijian();
92. fragmentTransaction.add(R.id.content, tuijian);
93. }
94. fragmentTransaction.show(tuijian);
95. break;
96. case 1:
97. if (daohang==null) {
98. new Daohang();
99. fragmentTransaction.add(R.id.content, daohang);
100. }
101. fragmentTransaction.show(daohang);
102. break;
103. case 2:
104. if (faxian==null) {
105. new Faxian();
106. fragmentTransaction.add(R.id.content, faxian);
107. }
108. fragmentTransaction.show(faxian);
109. break;
110. case 3:
111. if (wode==null) {
112. new Wode();
113. fragmentTransaction.add(R.id.content, wode);
114. }
115. fragmentTransaction.show(wode);
116.
117. break;
118. case 4:
119. if (lixian==null) {
120. new Lixian();
121. fragmentTransaction.add(R.id.content, lixian);
122. }
123. fragmentTransaction.show(lixian);
124. break;
125. }
126. fragmentTransaction.commit();
127. }
128.
129. private void ClearImage() {
130. tuijian_iv.setImageResource(R.drawable.tuijian1);
131. daohang_iv.setImageResource(R.drawable.daohang1);
132. faxian_iv.setImageResource(R.drawable.faxian1);
133. wode_iv.setImageResource(R.drawable.wode1);
134. lixian_iv.setImageResource(R.drawable.lixian1);
135. }
136.
137. class MyOnClickListener implements OnClickListener {
138.
139. @Override
140. public void onClick(View v) {
141. // TODO Auto-generated method stub
142. ClearImage();
143. switch (v.getId()) {
144. case R.id.tuijian_iv:
145. tuijian_iv.setImageResource(R.drawable.tuijian2);
146. 0);
147. break;
148. case R.id.daohang_iv:
149. daohang_iv.setImageResource(R.drawable.daohang2);
150. 1);
151. break;
152. case R.id.faxian_iv:
153. faxian_iv.setImageResource(R.drawable.faxian2);
154. 2);
155. break;
156. case R.id.wode_iv:
157. wode_iv.setImageResource(R.drawable.wode2);
158. 3);
159. break;
160. case R.id.lixian_iv:
161. lixian_iv.setImageResource(R.drawable.lixian2);
162. 4);
163. break;
164. }
165. }
166.
167. }
168. }
说明:
- android.app.Fragment 主要用于定义Fragment
- android.app.FragmentManager 主要用于在Activity中操作Fragment
- android.app.FragmentTransaction 保证一些列Fragment操作的原子性
- 获取FragmentManage的方式有:android.app.Fragment:getFragmentManager() ;android.support.v4.app.Fragment:getSupportFragmentManager
- FragmentTransaction transaction = fm.benginTransatcion()用来开启一个事务
- transaction.add()
往Activity中添加一个Fragment
transaction.remove()
从Activity中移除一个Fragment,如果被移除的Fragment没有添加到回退栈,这个Fragment实例将会被销毁;数据会被销毁。
transaction.replace()
使用另一个Fragment替换当前的,实际上就是remove()然后add()的合体。
transaction.hide();transaction.show()
隐藏当前的Fragment,仅仅是设为不可见,并不会销毁,显示之前隐藏的Fragment,数据不会销毁。
五 效果:
日志结果:
从日志输出中可以看到和上图的生命周期顺序是一至的,我上面使用的hide和show方法的,所以程序启动后不管怎么切换fragment都不会销毁fragment。