一 基本使用

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. MinGW的安装

    我在MinGW官网下载到的版本是mingw-w64-install.exe,不过这差不多是一年以前的事了……   安装路径:D:\Program Files (x86)\mingw-w64\i686- ...

  2. C#多播委托详解

    包含多个方法的委托成为多播委托,调用多播委托,可以按照顺序连续调用多个方法,因此,委托的签名就必须返回void;否则,就只能得到委托调用的最好一个方法的结果 1.多播委托可以用运算符"+&q ...

  3. oracle数据库中的存储过程

    存储过程是一组为了完成特定功能的sql语句集,是一段sql代码片段,经编译后存储在数据库中,用户通过指定存储过程的名字并给出参数(如果存储过程存在参就给出,不存在就不用给出参数)来执行它.因为它是一段 ...

  4. spark 在启动的时候出现JAVA_HOME not set

    解决方法:在sbin目录下的spark-config.sh 中添加对应的jdk 路径,然后使用scp -r 命令复制到各个worker节点

  5. TreeView详细用法

    Treeview用于显示按照树形结构进行组织的数据.          Treeview控件中一个树形图由节点(TreeNode)和连接线组成.TtreeNode是TTreeview的基本组成单元. ...

  6. IP电话的配置

    内容描述:IP电话配置 问题描述: IP电话站点为8203,IP地址为10.11.6.3,电话状态为空心(不正常). 处理过程: 1.在浏览器中打开输入原先已经配置正常的IP话机的IP地址访问其配置, ...

  7. 矩阵快速幂 求斐波那契第N项

    #include<cstdio> #include<algorithm> #include<cstring> #include<iostream> us ...

  8. 10年前文章_mpc8313的ltib安装以及u-boot重新编译

    Linux系统下安装ltib(linux target image builder): 1.       下载光盘到本地 wget http://192.168.1.4/share/vendor/mp ...

  9. bzoj5017 [Snoi2017]炸弹 (线段树优化建图+)tarjan 缩点+拓扑排序

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=5017 题解 这个题目方法挺多的. 线段树优化建图 线段树优化建图的做法应该挺显然的,一个炸弹能 ...

  10. oracle date函数

    常用的时间格式 在oracle中有 yyyy-mm-dd hh24:mi:ss  而在Java中有些区别 为yyyy-MM-dd HH:mm:ss 这点还是经常容易模糊的.相信很多人都有过统计某些数据 ...