多应用+插件架构,代码干净,二开方便,首家独创一键云编译技术,文档视频完善,免费商用码云13.8K 广告
Fragment: 碎片,是一种可以嵌在活动中的UI片段. **1.静态:**在布局文件定义一个 ~~~ <fragment android:id="@+id/fragment1" android:name="com.example.gp08_day23_fragment2.fragment.TestFragment" android:layout_width="wrap_content" android:layout_height="wrap_content" /> //获取碎片管理器----可以直接在 activity获取到 manager = getFragmentManager(); //使用碎片管理器得到需要的碎片对象 Fragment fragment = manager.findFragmentById(R.id.fragment1); //得到碎片对象中的布局对象 View view = fragment.getView(); ~~~ **2.动态:** —-FragmentManager —-事务的使用 —-activity给fragment传值(Bundle对象) **2.1动态添加fragment步骤:** 1)在布局文件中给fragment占个位,一般用FrameLayout 2)创建fragment实例 3)获取到FragmentManager,在活动中可以直接调用getFragmentManager() 4)开启一个事务,通过调用beginTransaction方法开启 5)向容器内加入fragment,一般使用replace()方法实现,需要传入容器的id(就是在布局文件中占的位置)和fragment实例 6)提交事务,调用commit()方法完成。 **2.2先介绍一下传值** ~~~ 传值: //使用碎片管理器对象得到事务对象 FragmentTransaction transaction = manager.beginTransaction();//开启事务 // Fragment f = manager.findFragmentByTag("flag"); // if(f==null) // { // TestFragment fragment = new TestFragment(); // transaction.add(R.id.container, fragment, "flag"); // } TestFragment fragment = new TestFragment(); //activity给fragment传参数 Bundle bundle = new Bundle(); bundle.putString("msg", "hehe"+new Date()); fragment.setArguments(bundle); //replace是替换掉之前的fragment对象(remove--->add) transaction.replace(R.id.container, fragment); //提交事务 transaction.commit();//事务只能提交一次 接收: public class TestFragment extends Fragment { @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { TextView textView = new TextView(getActivity()); textView.setTextSize(20); //获取传递过来的参数 Bundle bundle = getArguments(); String value = bundle.getString("msg"); textView.setText(value); return textView; } } ~~~ 关键是这个: //activity给fragment传参数 Bundle bundle = new Bundle(); bundle.putString(“msg”, “hehe”+new Date()); fragment.**setArguments(bundle);** **3.生命周期** Fragment的生命周期:11个方法 * 1):初始化:对应于 activity的onCreate() * onAttach(Activity) 和所属的activity关联 * onCreate(Bundle )fragment的初始化 * onCreateView()初始化fragment显示的UI视图 * onActivityCreated()当activity的 onCreate()方法执行完(说明 activity的初始化已经完成) * 2):显示,隐藏 * onStart(),onResume(),onPause(),onStop() * 3):销毁 * onDestroyView() 销毁fragment显示的UI视图(View 对象) * onDestroy() 销毁的是fragment * onDettach() 和所属的activity断开 ![这里写图片描述](https://box.kancloud.cn/2016-04-20_571714ed64f3e.jpg "") 实例:源码地址见后面 和Activity联动: 创建时:从外到内创建 ![这里写图片描述](https://box.kancloud.cn/2016-04-20_5717152972840.jpg "") 关闭时:从内到外销毁 ![这里写图片描述](https://box.kancloud.cn/2016-04-20_5717152984a5c.jpg "") **4.自定义模拟回退栈** ~~~ // 作为回退栈 private LinkedList<Fragment> stack = new LinkedList<Fragment>(); public void addFragment(View v) { TestFragment fragment = new TestFragment(); Bundle bundle = new Bundle(); bundle.putString("msg", "hahahaha" + new Date()); fragment.setArguments(bundle); // 如果栈为空直接入栈,不为空则把当前的activity隐藏,然后加入新的fragment if (stack.size() == 0) getFragmentManager().beginTransaction() .add(R.id.container, fragment).commit(); else getFragmentManager().beginTransaction().hide(stack.peek())// 隐藏栈顶的 .add(R.id.container, fragment).commit(); // 入栈 stack.push(fragment); } // 点击回退按钮显示上一个fragment public void backFragment(View v) { if (!stack.isEmpty()) { getFragmentManager().beginTransaction().remove(stack.poll()) .commit(); } if (!stack.isEmpty()) getFragmentManager().beginTransaction().show(stack.peek()).commit(); } ~~~ 效果图: ![这里写图片描述](https://box.kancloud.cn/2016-04-20_5717152996c93.jpg "") **5.使用baskToStack(null);方法** ~~~ public void addFragment(View v) { TestFragment fragment = new TestFragment(); Bundle bundle = new Bundle(); bundle.putString("msg", "xixi" + new Date()); fragment.setArguments(bundle); FragmentTransaction tran = getFragmentManager().beginTransaction(); tran.add(R.id.container, fragment);//remove1 add 2 tran.addToBackStack(null);//把事务加入当前的回退栈 回滚 tran.commit(); } // 点击回退按钮显示上一个fragment public void backFragment(View v) { onBackPressed(); } ~~~ 效果和自定义回退栈相同。 –>使用回退栈的作用主要是返回时返回到上一个fragment **6.横屏竖屏动态变化布局** 1)创建一个在横屏时加载的布局文件夹 ![这里写图片描述](https://box.kancloud.cn/2016-04-20_57171529dc498.jpg "") 竖屏布局:只有一个fragment ~~~ <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context=".MainActivity" > <fragment android:id="@+id/fragment" android:layout_width="200dp" android:layout_height="match_parent" android:name="com.beiing.fragmentlandscapefragment.FilesFragment" /> </RelativeLayout> ~~~ 横屏布局: ~~~ <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context=".MainActivity" > <fragment android:id="@+id/fragment" android:layout_width="200dp" android:layout_height="match_parent" android:name="com.beiing.fragmentlandscapefragment.FilesFragment" /> <FrameLayout android:id="@+id/container" android:layout_width="match_parent" android:layout_height="match_parent" android:layout_toRightOf="@id/fragment" /> </RelativeLayout> ~~~ 2)判断横屏竖屏:根据横屏竖屏得到不同的操作 ~~~ // 判断屏幕方向 if (getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE) { ContentFragment fragment = new ContentFragment(); fragment.setArguments(bundle); getFragmentManager().beginTransaction() .replace(R.id.container, fragment) .addToBackStack(null) .commit(); } else { Intent intent = new Intent(getActivity(),ContentActivity.class); intent.putExtras(bundle); startActivity(intent); } ~~~ 效果图: ![这里写图片描述](https://box.kancloud.cn/2016-04-20_5717152a013df.jpg "") **我的总结:** 1)实际开发中,fragment使用非常多,用法大都是在一个显示界面中底部放上一些按钮,中间一个fragment,这样就不用一个activity一个activity的跳了 2)引包的时候注意,一个是android.app.Fragment,还有一个向下兼容的 android.support.v4.app.Fragment;用后一个包的fragment时activity也要用这个包的FragmentActivity 3)在实现1)说的那种类似标签页时,需要注意不要每次都通过事务FragmentTransaction的replace()方法去变换界面,这样会导致该fragment重新创建,如果这个fragment里面有从网络获取数据的操作,这样,变换一次加载一次,谁受得了,解决方法是,在MainActivity中定义相应fragment的成员,然后一次加入FragmentManager中,然后想要显示哪个fragment只用调用FragmentTransaction 的show方法和hide方法。后面具体项目会遇到。 [源码](http://download.csdn.net/detail/u011102153/9093565)