一 基本使用

Fragment依附于Activity使用,方面我们在一个页面里面切换显示多屏内容。

Activity管理Fragment有两种方式,通过FragmentTransacation这个类来管理fragment的显示、隐藏

1 replace方法

 FragmentTransaction transaction = mBaseActivity.getSupportFragmentManager().beginTransaction();
transaction.replace(R.id.fl_container, mCurrent).commit();

使用这种方法,可以避免出现fragment重叠问题。但每次执行replace时,fragment都会重新走一遍生命周期方法,会造成重复加载数据。不推荐使用。

2 hide,show,add方法

     private void switchFragment(Fragment from, Fragment to) {
if (mCurrent != to) {
mCurrent = to;
FragmentTransaction transaction = mBaseActivity.getSupportFragmentManager().beginTransaction();
// 判断目标fragment是否被add过
if (!to.isAdded()) {
transaction.hide(from).add(R.id.fl_container, to,to.getClass().getName()).commit();
} else {
transaction.hide(from).show(to).commit();
}
}
}

使用这种方法,避免了重新加载数据问题,但是会出现fragment重叠问题。

原因:当内存不足时,系统会回收宿主Activity,而Fragment实例并没有被一并回收,activity被回收时会主动调用onSaveInstanceState方法,保存视图层(View Hierarchy)及一些activity的数据。

二 重叠问题

通过onCreate中的参数savedInstanceState恢复了之前的fragment。此时的FragmentTransaction中的相当于又再次add了fragment进去的,之前保存的fragment也还在。hide()和show()方法对之前保存的fragment已经失效了。之前显示的fragment一直显示,所以出现了重叠的现象。解决方法:

首先,activity被系统收回,触发onSaveInstanceState(Bundle outState)方法,在outstate里,系统自动保存了fragment数据,手动保存当前显示的fragment索引

    private static final String SAVED_CURRENT_ID = "currentId";
private int currentIndex;
@Override
public void onSaveInstanceState(Bundle outState, PersistableBundle outPersistentState) {
outState.putInt(SAVED_CURRENT_ID, currentIndex);//保存当前显示fragment索引
super.onSaveInstanceState(outState, outPersistentState);
}

其次,当重新进入activity时,走生命周期方法oncreae(Bundle savedInstanceState),saveInstanceState 包含了fragment数据,以及其他一些view的状态,和我们手动保存的数据,从bundle里取出fragment对象,避免重新实例化fragment,造成视图重叠

     @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//重启时防止fragment重叠
if (savedInstanceState != null) {
currentIndex = savedInstanceState.getInt(SAVED_CURRENT_ID, 0);
homeFragment = (HomeFragment) getSupportFragmentManager().findFragmentByTag(HomeFragment.class.getName());
newsFragment = (NewsFragment) getSupportFragmentManager().findFragmentByTag(NewsFragment.class.getName());
findFragment = (FindFragment) getSupportFragmentManager().findFragmentByTag(FindFragment.class.getName());
adminFragment = (AdminFragment) getSupportFragmentManager().findFragmentByTag(AdminFragment.class.getName());
} else {
initFragment();
}
intView();
} @Override
protected void intView() {
setTitle("主页");
radioGroup = (RadioGroup) findViewById(R.id.rg_main_bottom);
radioGroup.setOnCheckedChangeListener(occl);
((RadioButton) radioGroup.getChildAt(currentIndex)).setChecked(true);
}

最后,恢复选中fragment

     private RadioGroup.OnCheckedChangeListener occl = new RadioGroup.OnCheckedChangeListener() {

         @Override
public void onCheckedChanged(RadioGroup group, int checkedId) {
switch (checkedId) {
case R.id.rb_home:
setTitle("首页");
currentIndex = 0;
if (homeFragment == null) {
homeFragment = HomeFragment.newInstance(null, null);
}
switchFragment(mBackHandledFragment, homeFragment);
break;
case R.id.rb_news:
setTitle("新闻");
currentIndex = 1;
if (newsFragment == null) {
newsFragment = NewsFragment.newInstance(null, null);
}
switchFragment(mBackHandledFragment, newsFragment);
break;
case R.id.rb_find:
showTabSelectView();
currentIndex = 2;
if (findFragment == null) {
findFragment = FindFragment.newInstance(null, null);
}
switchFragment(mBackHandledFragment, findFragment);
break;
case R.id.rb_admin:
setTitle("我的");
currentIndex = 3;
if (adminFragment == null) {
adminFragment = AdminFragment.newInstance(null, null);
}
switchFragment(mBackHandledFragment, adminFragment);
break;
}
}
}; private void switchFragment(Fragment from, Fragment to) {
FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
if (null == from) {
if (mBackHandledFragment != to) {
mBackHandledFragment = (BackHandledFragment) to;
if (!to.isAdded()) {
transaction.add(R.id.frameLayout_container, to, to.getClass().getName()).commit();
} else {
transaction.show(to).commit();
}
}
} else {
if (mBackHandledFragment != to) {
mBackHandledFragment = (BackHandledFragment) to;
// 判断目标fragment是否被add过
if (!to.isAdded()) {
transaction.hide(from).add(R.id.frameLayout_container, to, to.getClass().getName()).commit();
} else {
transaction.hide(from).show(to).commit();
}
}
}
}

Fragment 基础使用及重叠问题的更多相关文章

  1. 札记:Fragment基础

    Fragment概述 在Fragment出现之前,Activity是app中界面的基本组成单位,值得一提的是,作为四大组件之一,它是需要"注册"的.组件的特性使得一个Activit ...

  2. Fragment基础----信息传递

    如何通过activity来访问fragment的信息呢,我们可以通过一个实例来了解. 使用两个Fragment和一个activity管理 效果图: 第一个fragment来表示输入框 第二个fragm ...

  3. Fragment基础----生命周期

    Fragment生命周期和Activity对比 注意:在一个app的运行期间,前台的activity有时可能会被其他的视图组件打断,然后进入pause状态. 比如打开一个半透膜的activity (比 ...

  4. Fragment基础----创建

    1,Fragment的目的及应用场景 fragment 是3.0后引入的类,其字面翻译为“碎片”. 目的是将activity划分成许多单元再进行组合,可以根据不同分辨率屏幕,在不同状态下,灵活创建优化 ...

  5. android之Fragment基础详解(一)

      一.Fragment的设计哲学 Android在3.0中引入了fragments的概念,主要目的是用在大屏幕设备上--例如平板电脑上,支持更加动态和灵活的UI设计.平板电脑的屏幕比手机的大得多,有 ...

  6. Fragment基础讲解

    //新建一个碎片public class LeftFragment extends Fragment { @Override public View onCreateView(LayoutInflat ...

  7. Android Studio 单刷《第一行代码》系列 05 —— Fragment 基础

    前情提要(Previously) 本系列将使用 Android Studio 将<第一行代码>(书中讲解案例使用Eclipse)刷一遍,旨在为想入坑 Android 开发,并选择 Andr ...

  8. Android Fragment基础及使用

    同一个app内的界面切换 用Fragment比较合适,因为Activity比较重量级 Fragment 轻量级,切换灵活 --------------------------------------- ...

  9. Fragment 基础

    FragmentActivity与Activity区别 fragment是3.0以后的东西,为了在低版本中使用fragment就要用到android-support-v4.jar兼容包,而fragme ...

随机推荐

  1. Python 入门之 递归

    Python 入门之 递归 1.递归: 递:一直传参 归:返回 (1)不断调用自己本身(无效递归 -- 死递归) def func(): print(1) func() func() (2)有明确的终 ...

  2. C#设计模式:解释器模式(Interpreter Pattern)

    一,C#设计模式:解释器模式(Interpreter Pattern) 1,解释器模式的应用场合是Interpreter模式应用中的难点,只有满足“业务规则频繁变化,且类似的模式不断重复出现,并且容易 ...

  3. scala学习笔记(7)

    1.包 --------------------------------------- Scala中的包和Java或者C++中命名空间的目的是相同的:管理大型程序中的名称. package a{ pa ...

  4. 要了解mysql原理,还是要心里有点B树才行

      要了解数据库索引的底层原理,我们就得先了解一种叫树的数据结构,而树中很经典的一种数据结构就是二叉树!所以下面我们就从二叉树到平衡二叉树,再到B-树,最后到B+树来一步一步了解数据库索引底层的原理! ...

  5. loadkeys - 调入键盘翻译表

    总览 (SYNOPSIS) loadkeys [ -d --default ] [ -h --help ] [ -q --quiet ] [ -v --verbose [ -v --verbose ] ...

  6. GUI学习之十八——QDateTimeEdit学习总结

    在前面两章我们总结了QSpinBox和QDoubleSpinBox的用法,今天来总结一下QDateTimeEdit控件的基本用法 一.描述 1.QDateTimeEdit是一个用来编辑日期和时间的单行 ...

  7. pyhive

    from pyhive import hiveimport pandas as pdimport numpy as npclass myhive():    def __init__(self,hos ...

  8. JVM 常量池、运行时常量池、字符串常量池

    常量池: 即class文件常量池,是class文件的一部分,用于保存编译时确定的数据. 保存的内容如下图: D:\java\test\out\production\test>javap -ver ...

  9. python中strftime和strptime函数

    strftime和strptime函数均来自包datetime from datetime import * strftime: 将datetime包中的datetime类,按照入参格式生成字符串变量 ...

  10. axios中put和patch的区别(都是update , put是需要提交整个对象资源,patch是可以修改局部)

    patch方法用来更新局部资源,这句话我们该如何理解? 假设我们有一个UserInfo,里面有userId, userName, userGender等10个字段.可你的编辑功能因为需求,在某个特别的 ...