一 基本使用

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. 18: vue-element-admin使用

    1.1 vue-element-admin使用 1.使用参考网站 1)官方演示环境: https://panjiachen.github.io/vue-element-admin/#/dashboar ...

  2. linux:服务器代理squid安装配置

    国内上往外的网站太慢,配了个香港代理服务器.如下:当前环境: centos系统.香港服务器IP(假设:59.188.71.11)检查squid是否安装:[root@localhost ~]# rpm ...

  3. 剑指offer-二叉搜索树的第k个结点树-python

    二叉查找树(Binary Search Tree),(又:二叉搜索树,二叉排序树)它或者是一棵空树,或者是具有下列性质的二叉树: 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值: 若它的 ...

  4. 虚拟机的linux(CentOS)系统安装

    一.安装虚拟机:VirtualBox  1.软件下载:https://www.virtualbox.org/  2.一路点击下一步完成安装 二.安装linux:CentOS6.8  1.官方网站:ht ...

  5. 封装class类--不分割类名

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  6. /etc/nscd.conf - 域名服务缓存守护进程配置文件

    描述 DESCRIPTION 该文件 /etc/nscd.conf 在启动 nscd(8) 时读入.每一行或者指定一个属性和值,或者指定一个属性.服务和一个值.域之间通过空格或者TAB分开.‘#’表示 ...

  7. 02java基础——类和方法

    1.类的定义 /* 定义类: 使用类的形式,对现实中的事物进行描述 事物: 属性,方法 属性: 变量 方法: 这个事物具备的功能 格式: public class 类名{ 属性定义 修饰符 数据类型 ...

  8. 费用流+SPFA ||【模板】最小费用最大流

    题面:[模板]最小费用最大流 代码: #include<cstdio> #include<cstring> #include<iostream> #include& ...

  9. u-boot-2016.09 make编译过程分析(一)

    https://blog.csdn.net/guyongqiangx/article/details/52565493 综述 u-boot自v2014.10版本开始引入KBuild系统,Makefil ...

  10. index 索引

    1.创建表 drop table if exists kg_fk_user;create table kg_fk_user(id int,name string)row format delimite ...