用Fragments在Android中进行多窗格开发

2024-09-03 04:18

本文主要是介绍用Fragments在Android中进行多窗格开发,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文翻译于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 {@Overridepublic 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;@Overridepublic 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() {@Overridepublic void onClick(View v) {updateDetail();}});return view;}public interface OnItemSelectedListener {public void onRssItemSelected(String link);}@Overridepublic void onAttach(Activity activity) {super.onAttach(activity);if (activity instanceof OnItemSelectedListener) {listener = (OnItemSelectedListener) activity;} else {throw new ClassCastException(activity.toString()+ " must implemenet MyListFragment.OnItemSelectedListener");}}@Overridepublic void onDetach() {super.onDetach();listener = null;}// may also be triggered from the Activitypublic void updateDetail() {// create a string just for testingString newTime = String.valueOf(System.currentTimeMillis());// inform the Activity about the change based// interface defintionlistener.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来代替。

这篇关于用Fragments在Android中进行多窗格开发的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1131997

相关文章

C#使用SQLite进行大数据量高效处理的代码示例

《C#使用SQLite进行大数据量高效处理的代码示例》在软件开发中,高效处理大数据量是一个常见且具有挑战性的任务,SQLite因其零配置、嵌入式、跨平台的特性,成为许多开发者的首选数据库,本文将深入探... 目录前言准备工作数据实体核心技术批量插入:从乌龟到猎豹的蜕变分页查询:加载百万数据异步处理:拒绝界面

Android中Dialog的使用详解

《Android中Dialog的使用详解》Dialog(对话框)是Android中常用的UI组件,用于临时显示重要信息或获取用户输入,本文给大家介绍Android中Dialog的使用,感兴趣的朋友一起... 目录android中Dialog的使用详解1. 基本Dialog类型1.1 AlertDialog(

Python使用自带的base64库进行base64编码和解码

《Python使用自带的base64库进行base64编码和解码》在Python中,处理数据的编码和解码是数据传输和存储中非常普遍的需求,其中,Base64是一种常用的编码方案,本文我将详细介绍如何使... 目录引言使用python的base64库进行编码和解码编码函数解码函数Base64编码的应用场景注意

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

Python基于wxPython和FFmpeg开发一个视频标签工具

《Python基于wxPython和FFmpeg开发一个视频标签工具》在当今数字媒体时代,视频内容的管理和标记变得越来越重要,无论是研究人员需要对实验视频进行时间点标记,还是个人用户希望对家庭视频进行... 目录引言1. 应用概述2. 技术栈分析2.1 核心库和模块2.2 wxpython作为GUI选择的优

Java进行文件格式校验的方案详解

《Java进行文件格式校验的方案详解》这篇文章主要为大家详细介绍了Java中进行文件格式校验的相关方案,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、背景异常现象原因排查用户的无心之过二、解决方案Magandroidic Number判断主流检测库对比Tika的使用区分zip

Java使用Curator进行ZooKeeper操作的详细教程

《Java使用Curator进行ZooKeeper操作的详细教程》ApacheCurator是一个基于ZooKeeper的Java客户端库,它极大地简化了使用ZooKeeper的开发工作,在分布式系统... 目录1、简述2、核心功能2.1 CuratorFramework2.2 Recipes3、示例实践3

基于Flask框架添加多个AI模型的API并进行交互

《基于Flask框架添加多个AI模型的API并进行交互》:本文主要介绍如何基于Flask框架开发AI模型API管理系统,允许用户添加、删除不同AI模型的API密钥,感兴趣的可以了解下... 目录1. 概述2. 后端代码说明2.1 依赖库导入2.2 应用初始化2.3 API 存储字典2.4 路由函数2.5 应

利用Python开发Markdown表格结构转换为Excel工具

《利用Python开发Markdown表格结构转换为Excel工具》在数据管理和文档编写过程中,我们经常使用Markdown来记录表格数据,但它没有Excel使用方便,所以本文将使用Python编写一... 目录1.完整代码2. 项目概述3. 代码解析3.1 依赖库3.2 GUI 设计3.3 解析 Mark

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda