本文翻译于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。
第二种途径是最灵活的,通常也是使用fragment的最好方式。在这样的情况下main activity检查在布局中detail fragment是否是可用的。如果detailed fragment存在,mian activity会告诉fragment,它应当自己更新自己。如果detail fragment不可用,main activity就启动detailed 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来代替。

更多相关文章

  1. Android-工作遭遇-URLConnection原生请求http和https忽略证书
  2. Android之-----GridView使用的方法总结
  3. Android(安卓)Handler那些事儿(二)——几个关键类之间的关系
  4. android实现一个简单的加法功能
  5. Flutter(三):实现Flutter代码调用Android原生代码(创建WebView Plu
  6. Android事件处理之使用异步任务执行下载
  7. Android属性动画完全解析(中),ValueAnimator和ObjectAnimator的高
  8. Android(安卓)Context简介
  9. Android之ListActivity(一):布局与数据绑定

随机推荐

  1. 我的android 第一天-电话拨号器
  2. Android:shape的使用详解(1)
  3. android的binder机制研究二
  4. Android(安卓)中级教程之------Android(
  5. Android、iOS系统架构
  6. android 上 webkit js 扩展之全局本地对
  7. Android(安卓)安全机制概述 Permission
  8. Android的消息处理机制(深入源码)
  9. android EditText inputType 及 android:
  10. freetype 在android编译时上的一个makefi