本文翻译于vogella.com。
在Android应用程序中使用Fragments
这篇教程描述了怎样在Android 应用程序中使用Fragment类来创建多窗格布局,也就是可以缩放到设备的可用宽度的应用程序。它基于Eclipse 4.3(Kepler),Java1.6和Android4.4。
1.Android基础
下面的描述假设了你已经掌握了基本的Android开发知识。
请查看Android开发教程来学习基础知识。也可以看Android开发教程来获取更多的关于Android开发的信息。
2.Fragments
2.1 什么是Fragments?
fragment是一个可以被Activity使用的独立的部件。Fragment将功能进行封装所以它更见易于被重用在actvities和layouts之中。
fragment运行在一个Activity的Context中,但拥有它自己的生命周期以及作为特色的用户界面。也可以定义没有用户界面的fragment,也就是headless fragment。
Fragment可以被动态或者静态地添加到一个Activity中。
2.2 使用fragment的好处
Fragment使在不同的布局中重用部件变的容易,例如,你可以为创建单窗格的手机布局和多窗格的平板布局。这没有限制到平板设备上;例如,你也可以使用fragment在智能手机上来支持横屏和竖屏的不同布局。
典型的例子是一个Activity中展示item的一个列表。在平板设备上如果你在item上点击,你可以立即在同一块屏幕的右侧看到item的详情。在智能手机上你会跳转到一个新的详情界面。下面的图片就描述了这些。
接下来的讨论将假设你有两个fragment(main和detail),你也可以有更多。我们将会有一个main activity和一个detailed activity。在平板上main activity的布局中包含两个fragment,在手持设备上只包含main fragment。
下面的截图表明了这样的用法。
2.3 如何使用fragment
用fragment创建不同的布局,你可以:
- 使用一个activity,平板上显示两个fragment。在这种情况下你将在任何必要的时候在activity中切换fragment。这就需要fragment不能声明在layout文件中因为那样的fragment不能在运行期间被移除。
- 在手机上使用各自的activity来持有每一个fragment。例如,但平板电脑UI在一个activity中使用两个fragment的时候,在手机上使用同一个activity,但提供一个可供选择的仅包含一个fragment的布局。当你需要切换fragment,启动另一个持有其他fragment的activity。
3. fragment生命周期
fragment的生命周与持有它的activity的生命周期相关联。
表 1. Fragment 生命周期
4. 定义和使用Fragment
4.1 定义Fragment
定义一个新的fragment,你要么继承android.app.fragment类要么一个它的子类,例如,ListFragment,DialogFragment,PreferenceFragment或者WebViewFragment。下面的代码展示了一种实现。
package com.example.android.rssfeed; import android.app.Fragment; import android.os.Bundle; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.TextView; public class DetailFragment extends Fragment { @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View view = inflater.inflate(R.layout.fragment_rssitem_detail, container, false); return view; } public void setText(String item) { TextView view = (TextView) getView().findViewById(R.id.detailsText); view.setText(item); } }
4.2 静态添加Fragment
使用你的新的fragment,你可以静态地将它添加到XML布局中。
检查fragment是否已经是你的布局的一部分,你可以使用FragmentManager类。
DetailFragment fragment = (DetailFragment) getFragmentManager(). findFragmentById(R.id.detail_frag); if (fragment==null || ! fragment.isInLayout()) { // start new Activity } else { fragment.update(...); }
如果fragment定义在XML布局文件中,那么android:name属性指向对应的类。
4.3 Fragment生命周期
Fragment有它自己的生命周期。但是它总是与用这个fragment的activity的生命周期有联系。
onCreate()方法在activity的onCreate方法之后但在fragment的onCreateView()方法之前被调用。
一旦fragment应该创建它的用户界面时,onCreateView()方法就被Android调用。这里你可以通过这个方法传递来的Inflator对象调用inflate()方法来inflate布局。在headless fragment中没有必要去实现这个方法。
当宿主activity创建完毕,在onCreateView()方法之后,onActivityCreate()方法就会被调用。这个方法里,你可以实例化那些需要Context对象的对象。
Fragment不是Context的子类,你必须使用getActivity()方法来得到parent activity。
一旦fragment可见了,onStart()方法将被调用。
如果一个activity停止了(stop),它的fragment也会停掉;如果一个activity被销毁了,它的fragment也就销毁掉了。
4.4 应用与fragment通信
为了增加fragment的重用,他们不应当直接与彼此通信。fragment的每一次通信的完成都应该通过它的宿主activity。
为了这个目的,fragment应该定义一个内部接口并要求那些用这个fragment的activity必须实现这个接口。这种方式下你就避免了fragment对用它的activity有任何认识。在它的onAttach()方法中可以检查activity是否正确地实现了这个接口。
例如,假设你有一个应该传递一个值给它的父activity的fragment。这样的情形可以像下面那样实现。
package com.example.android.rssfeed; import android.app.Activity; import android.app.Fragment; import android.os.Bundle; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.Button; public class MyListFragment extends Fragment { private OnItemSelectedListener listener; @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View view = inflater.inflate(R.layout.fragment_rsslist_overview, container, false); Button button = (Button) view.findViewById(R.id.button1); button.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { updateDetail(); } }); return view; } public interface OnItemSelectedListener { public void onRssItemSelected(String link); } @Override public void onAttach(Activity activity) { super.onAttach(activity); if (activity instanceof OnItemSelectedListener) { listener = (OnItemSelectedListener) activity; } else { throw new ClassCastException(activity.toString() + " must implemenet MyListFragment.OnItemSelectedListener"); } } @Override public void onDetach() { super.onDetach(); listener = null; } // may also be triggered from the Activity public void updateDetail() { // create a string just for testing String newTime = String.valueOf(System.currentTimeMillis()); // inform the Activity about the change based // interface defintion listener.onRssItemSelected(newTime); } }
5. fragment中的数据保持
5.1 保持应用重新启动之间的数据
在fragment中,你也需要保存你的应用数据。鉴于此你可以保持你的数据在一个中心位置。例如,
- SQLite数据库
- File文件
- Application对象,这种情况下,应用需要处理存储。
5.2 保持配置改变之间的数据
如果你想要保持数据在配置改变的时候,你也可以使用application对象。
另外,你也可以调用在fragment的setRetainState(true)方法。配置改变之间的这个保留的fragment实例只在fragment没有被添加到backstack才会有效。对那些有用户界面的fragment,Google不推荐使用这个方法。在这种情况下数据必须保存为成员变量。
如果Bundle类支持那些应该被保存的数据,你也可以使用onSaveInstanceState()方法来放置数据到Bundle里,在onActivityCreated()方法恢复这些数据。
6. 在运行时修改Fragment
FragmentManager和FragmentTransaction类可以让你添加,移除以及替换你的activity布局中的fragment。
Fragment可以通过事务被动态地修改。动态地添加一个fragment到一个已存在的布局中,你通常地在XML布局文件中定义一个容器,在其中添加你的fragment。为了达到这样的效果你可以使用,例如,一个FrameLayout元素。
FragmentTransaction ft = getFragmentManager().beginTransaction(); ft.replace(R.id.your_placehodler, new YourFragment()); ft.commit();
一个新的Fragment将会替换掉之前添加到容器里的已存在的Fragment。
如果你想添加事务到Android的backstack,你可以使用addToBackStack()方法。这将会添加动作到activity的历史stack中,也就是这将允许通过返回键还原Fragment的改变。
7. Fragment过渡动画
在fragment事务处理间你可以通过setCustonAnimations()方法定义应该被基于属性动画API使用的动画。
你也可以通过setTransition()方法的调用使用几个由Android提供的标准的动画。这些通过常量开头于FragmentTransation.TRANSITION_FRAGMENT_*的方式定义的。
两个方法都允许你来定义一个条目动画和一个已存在的动画。
8. 添加Fragment事务到backstack
你可以添加一个FragmentTransaction到backstack来让用户使用返回键来还原(或回滚)事务。
为了达到此目的你可以使用FragmentTransaction对象的addToBackStack()方法。
9. 进行后台处理的Fragment
9.1 Headless Fragment
Fragment可以不定义用户界面来使用。
来实现一个headless fragment,简单地在你的fragment中onCreateView()方法中返回null就行。
Tip:推荐联合着setRetainInstance方法来使用headless fragment进行后台处理。这种方式下你在你的异步处理期间你自己不必处理配置变化。
9.2 保留的headless fragment来处理配置改变
Headless fragment通常用来在经历配置改变的时候封装一些数据或者后台处理任务。为了这样的目的你应将你的headless fragment设置成保留的。一个保留的fragment在配置改变期间不会被销毁。
设置你的fragment被保留,调用它的setRetainInstance()方法。
添加这样的fragment到一个activity中,你要使用FragmentManager类中的add()方法。如果你以后需要找到这个Fragment,你需要给它添加一个tag来做到可以用FragmentManger类的findFragmentByTag()方法查找到。
注意:onRetainNonConfigurationInstance()的用法已经弃用了,应该用保留的headless fragment来代替。