热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Android应用UI开发中Fragment的常见用法小结

这篇文章主要介绍了Android应用UI开发中Fragment的常见用法小结,Fragment的存在是为了解决不同屏幕分辩率的动态和灵活UI设计,需要的朋友可以参考下

1.Fragment概述
在一个Activity中, Fragment代表UI的一个部分或者一个行为。一个Activity可以结合多个Fragment对象,也可以在多个activity中使用相同Fragment字节码对应的不同对象。一个Fragment对象必须被嵌入在一个主Activity对象中,该Fragment的生命周期与主Activity息息相关。比如,当主Activity处于paused状态,其对应的所有Fragment对象均处于paused状态,只有当主Activity处于resumed状态时,Fragment才能处于自由控制状态。


2.创建Fragment
为了创建一个Fragment,应该去继承Fragment或者其子类,覆写相应的方法。比如onCreate(),OnCreateView(),onPause()等等
(1).添加UI界面
为该Fragment展现一个布局,必须去实现onCreateView()回掉方法。
注意:当该Fragment继承了ListFragment时,不需要覆写onCreateView()方法,因为默认返回一个ListView对象

public View onCreateView(LayoutInflater inflater, ViewGroup container, 
  Bundle savedInstanceState) { 
 View view = inflater.inflate(R.layout.list, null); 
 return view; 
} 

(2).添加Fragment到Activity
1).通过layout布局文件
android:name属性应该为Fragment对应类的完整路径。

<&#63;xml version="1.0" encoding="utf-8"&#63;> 
 
 
  
  
 
 

2).通过Java代码
当Activity运行时,可以自由的在该activity上添加fragment对象,但应该指定一个ViewGroup容器,可以FragmentTransaction完成fragment的添加移除或者替换。

manager = getFragmentManager(); 
if(manager.findFragmentByTag("right") == null){ 
 manager.beginTransaction().replace(R.id.right, new RightFrag(), "right").commit(); 
} 

(3).fragment唯一标示符
每个fragment需要定义一个唯一的标识符,如果activity被销毁又重新启动,系统能够恢复该fragment的状态。如果想重新恢复,需满足下面有3种方式之一:
1).定义ID
在布局文件中,定义android:id属性

 

2).指明tag
android:tag 指明 或者 一个fragment对象add()或者replace()时指定tag

 

或者

manager.beginTransaction() 
 .replace(R.id.right, new RightFrag(), "right")//在事务中指明该fragment的tag 
.commit(); 

3).viewgroup ID
如果该fragment均没有id和tag,系统将使用container view布局的id


3.Fragment的管理
通过getFragmentManager()方法,可以得到FragmentManager对象,主要完成下面的功能

FragmentManager manager = getFragmentManager(); 
(1).得到已经存在Fragment对象
如果该fragment在布局文件中指定了id,通过findFragmentById()得到对象,或者指定了tag可以通过findFragmentByTag()得到对象

Fragment fragment = getFragmentManager().findFragmentByTag("right"); 
//or 
Fragment fragment = getFragmentManager().findFragmentById(id); 

(2).注册OnBackStackChangedListener监听器
可以用来监听该任务对应的返回栈信息,当该返回栈状态发生改变时,执行对应的onBackStackChanged() 方法

manager.addOnBackStackChangedListener(new FragmentManager.OnBackStackChangedListener() { 
 @Override 
 public void onBackStackChanged() { 
  Toast.makeText(MainActivity.this, "返回堆状态发生改变", 1).show(); 
 } 
}); 

(3).弹出返回栈
模拟用户点击返回键,将指定的fragment从返回栈中弹出,该操作为异步的。前提是该fragment对象使用.beginTransaction().addToBackStack("right")添加了进返回栈

manager.popBackStack(); //Pop the top state off the back stack 
(4).FragmentTransaction事务
事务主要包含一些操作的集合,比如增加移除替换,动画设置等等

/* 
 * 通过manager开启一个事务,该事务包含一些操作的集合,通事务可以 add(), remove(), replace() 
 * 完成对Fragment的操作,并使用commit()提交 
 */ 
FragmentTransaction transaction = manager.beginTransaction(); 
 
transaction.replace(R.id.right, new RightFrag(), "right"); 
transaction.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);//设置动画 
transaction.addToBackStack("right"); // 将该fragment加入返回堆 
// 提交事务 
transaction.commit(); 

(5).Fragment状态管理

/* 
 * 管理Fragment的状态 
 * 如果在一个主activityViewGroup中添加一个fragment, 
 * 如果手机屏幕旋转了,当前activity被销毁重建,fragment也被activityManager创建 
 * 故在onCreate中,需要判断一下 
 */ 
@Override 
protected void onCreate(Bundle savedInstanceState) { 
 super.onCreate(savedInstanceState); 
 setContentView(R.layout.activity_main); 
 manager = getFragmentManager(); 
 if (manager.findFragmentByTag("right") == null) { 
  // if(savedInstanceState == null)也可判断该fragment是否已经加载 
 
  manager.beginTransaction() 
   .replace(R.id.right, new RightFrag(), "right") 
   .setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN)// 设置动画 
   .addToBackStack("right") // 将该fragment加入返回堆 
  // 提交事务 
  .commit(); 
 } 
} 


4.Fragment间信息交互
(1).取得对象

/* 
 * 点击该Fragment的button按钮,将该button的text设置为另一个fragment中Edittext的文本值 
 */ 
public View onCreateView(LayoutInflater inflater, ViewGroup container, 
  Bundle savedInstanceState) { 
 View view = inflater.inflate(R.layout.list, null); 
 final Button button = (Button) view.findViewById(R.id.confirm); 
 button.setOnClickListener(new View.OnClickListener() { 
   
  @Override 
  public void onClick(View v) { 
   //通过FragmentManager找到另一个fragment中的edittext对象,并取得text内容 
   EditText editText = (EditText)(getFragmentManager().findFragmentByTag("left").getView().findViewById(R.id.name)); 
   button.setText(editText.getText().toString()); 
  } 
 }); 
 return view; 
} 

 (2).通回掉函数

public class MainActivity extends Activity { 
 private FragmentManager manager; 
 private Button button; 
 
 @Override 
 protected void onCreate(Bundle savedInstanceState) { 
  super.onCreate(savedInstanceState); 
  setContentView(R.layout.activity_main); 
   
  button.setOnClickListener(new View.OnClickListener() { 
   @Override 
   public void onClick(View v) { 
    RightFragment rightFrag = (RightFragment) (getFragmentManager().findFragmentByTag("right")); 
    /* 
     * 通过set方法,向其传递一个实例化对象,由于rightFrag.set()方法内部执行RightFragment.CallBack.get()方法,完成了参数的传递 
     */ 
    rightFrag.set(new RightFragment.CallBack() { 
     @Override 
     public void get(String str) { 
      button.setText(str); 
     } 
    }); 
   } 
  }); 
 } 
 
} 

public class RightFragment extends ListFragment { 
 private LoaderManager manager; 
 
 @Override 
 public void onCreate(Bundle savedInstanceState) { 
  super.onCreate(savedInstanceState); 
  manager = getLoaderManager(); 
 } 
 
 /* 
  * 点击该Fragment的button按钮,将该button的text设置为另一个fragment中Edittext的文本值 
  */ 
 public View onCreateView(LayoutInflater inflater, ViewGroup container, 
   Bundle savedInstanceState) { 
  View view = inflater.inflate(R.layout.list, null); 
  return view; 
 } 
 
 /** 
  * 通过调用该方法,接收一个回掉函数对象,callBack.get(str); 
  * @param callBack 
  */ 
 public void set(CallBack callBack) { 
  EditText editText = (EditText) getView().findViewById(R.id.name); 
  callBack.get(editText.getText().toString()); 
 } 
 
 /* 
  * 回掉接口 
  */ 
 interface CallBack { 
  public void get(String str); 
 } 
} 

5.FragmentManage:
FragmentManager能够实现管理activity中fragment. 通过调用activity的getFragmentManager()取得它的实例.
FragmentManager可以做如下一些事情:
(1)使用findFragmentById() (用于在activity layout中提供一个UI的fragment)或findFragmentByTag()
   (适用于有或没有UI的fragment)获取activity中存在的fragment
(2)将fragment从后台堆栈中弹出, 使用 popBackStack() (模拟用户按下BACK 命令).
(3)使用addOnBackStackChangeListener()注册一个监听后台堆栈变化的listener.
 
6.FragmentTransaction:
      FragmentTransaction对fragment进行添加,移除,替换,以及执行其他动作。
从 FragmentManager 获得一个FragmentTransaction的实例 :

FragmentManager fragmentManager = getFragmentManager(); 
FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();

每一个事务都是同时要执行的一套变化.可以在一个给定的事务中设置你想执行的所有变化,使用诸如 add(), remove(), 和 replace().然后, 要给activity应用事务, 必须调用 commit().

在调用commit()之前, 你可能想调用 addToBackStack(),将事务添加到一个fragment事务的back stack. 这个back stack由activity管理, 并允许用户通过按下 BACK 按键返回到前一个fragment状态.

举个例子, 这里是如何将一个fragment替换为另一个, 并在后台堆栈中保留之前的状态:

// Create new fragment and transaction 

Fragment newFragment = new ExampleFragment(); 

FragmentTransaction transaction = getFragmentManager().beginTransaction(); 

 // Replace whatever is in the fragment_container view with this fragment, 

// and add the transaction to the back stack transaction.replace(R.id.fragment_container, newFragment); transaction.addToBackStack(null); 

// Commit the transaction transaction.commit();

在这个例子中, newFragment 替换了当前layout容器中的由R.id.fragment_container标识的fragment.通过调用 addToBackStack(), replace事务被保存到back stack, 因此用户可以回退事务,并通过按下BACK按键带回前一个fragment.

如果添加多个变化到事务(例如add()或remove())并调用addToBackStack(), 然后在你调用commit()之前的所有应用的变化会被作为一个单个事务添加到后台堆栈, BACK按键会将它们一起回退.

添加变化到 FragmentTransaction的顺序不重要, 除以下例外:
必须最后调用 commit().
如果添加多个fragment到同一个容器, 那么添加的顺序决定了它们在view hierarchy中显示的顺序.
当执行一个移除fragment的事务时, 如果没有调用 addToBackStack(), 那么当事务提交后, 那个fragment会被销毁,并且用户不能导航回到它. 有鉴于此, 当移除一个fragment时,如果调用了 addToBackStack(), 那么fragment会被停止, 如果用户导航回来,它将会被恢复.

提示: 对于每一个fragment事务, 你可以应用一个事务动画, 通过在提交事务之前调用setTransition()实现.

调用 commit() 并不立即执行事务.恰恰相反, 它将事务安排排期, 一旦准备好, 就在activity的UI线程上运行(主线程).如果有必要, 无论如何, 你可以从你的UI线程调用 executePendingTransactions() 来立即执行由commit()提交的事务. 但这么做通常不必要, 除非事务是其他线程中的job的一个从属.

警告: 你只能在activity保存它的状态(当用户离开activity)之前使用commit()提交事务.

如果你试图在那个点之后提交, 会抛出一个异常.这是因为如果activity需要被恢复, 提交之后的状态可能会丢失.对于你觉得可以丢失提交的状况, 使用 commitAllowingStateLoss().


推荐阅读
  • 本文介绍了使用kotlin实现动画效果的方法,包括上下移动、放大缩小、旋转等功能。通过代码示例演示了如何使用ObjectAnimator和AnimatorSet来实现动画效果,并提供了实现抖动效果的代码。同时还介绍了如何使用translationY和translationX来实现上下和左右移动的效果。最后还提供了一个anim_small.xml文件的代码示例,可以用来实现放大缩小的效果。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文讲述了如何通过代码在Android中更改Recycler视图项的背景颜色。通过在onBindViewHolder方法中设置条件判断,可以实现根据条件改变背景颜色的效果。同时,还介绍了如何修改底部边框颜色以及提供了RecyclerView Fragment layout.xml和项目布局文件的示例代码。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • Android系统移植与调试之如何修改Android设备状态条上音量加减键在横竖屏切换的时候的显示于隐藏
    本文介绍了如何修改Android设备状态条上音量加减键在横竖屏切换时的显示与隐藏。通过修改系统文件system_bar.xml实现了该功能,并分享了解决思路和经验。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • docker增加restart=always, docker重启后自动启动容器的方法
    本文介绍了在运行docker容器时如何添加参数来保证每次docker服务重启后容器也自动重启的方法,以及如何使用命令来更新已启动的容器。 ... [详细]
  • 本文介绍了Hyperledger Fabric外部链码构建与运行的相关知识,包括在Hyperledger Fabric 2.0版本之前链码构建和运行的困难性,外部构建模式的实现原理以及外部构建和运行API的使用方法。通过本文的介绍,读者可以了解到如何利用外部构建和运行的方式来实现链码的构建和运行,并且不再受限于特定的语言和部署环境。 ... [详细]
  • 本文介绍了Java的集合及其实现类,包括数据结构、抽象类和具体实现类的关系,详细介绍了List接口及其实现类ArrayList的基本操作和特点。文章通过提供相关参考文档和链接,帮助读者更好地理解和使用Java的集合类。 ... [详细]
author-avatar
张淑男尧珊肇玲
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有