我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Android中Fragment管理及重叠问题的解决方法

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Android中Fragment管理及重叠问题的解决方法

一、Fragment介绍

fragment在3.0被引入以后,项目使用fragment越来越多,特别是主界面是底部tab页点击切换更换内容,当然啦, Fragment 在项目中存在着广泛的时候,例如通常在首页的设计中,通常底部的每一个 navigation 都对应这一个对应的 Fragment ,使用 Fragment 减轻了对应 Activity 的职责,让 Fragmen t充当了部分的Activity的职责。而且使用 Fragment 的时候,提高了代码和布局的封装和复用,这个优势也是特别的明显。

Fragment 拥有自己的生命周期管理,但是它是依赖对应的Activity的。

嗯,生命周期的介绍不是本篇的重点,贴个图加深一下印象。

二、Fragment的栈管理及其生命周期

addToShow


FragmentTransaction transaction = manager.beginTransaction();
 String tag = to.getClass().getSimpleName();
 transaction.add(from.getContainerId(), to, tag)
   .addToBackStack(tag)
   .hide(from)
   .show(to)
   .commit();

如果使用

add() 
hide() 
来控制跳转的话,对应的生命周期是这样的:


E/TAG: onAttach: Fragment09
 E/TAG: onCreate: Fragment09
 E/TAG: onCreateView: Fragment09
 E/TAG: onStart: Fragment09
 E/TAG: onResume: Fragment09
 E/____TAG____: onClick: 2131558527
 E/TAG: onAttach: Fragment10
 E/TAG: onCreate: Fragment10
 E/TAG: onHiddenChanged: Fragment09不可见了!
 E/TAG: onCreateView: Fragment10
 E/TAG: onStart: Fragment10
 E/TAG: onResume: Fragment10

如果此时从第二个 Fragment 再次返回到第一个 Fragment :


E/TAG: onHiddenChanged: Fragment10不可见了!
 E/TAG: onHiddenChanged: Fragment09可见了!!
 E/TAG: onPause: Fragment10
 E/TAG: onStop: Fragment10
 E/TAG: onDestroyView: Fragment10
 E/TAG: onDestroy: Fragment10
 E/TAG: onDetach: Fragment10

可以对应上面的图片,当返回的时候是直接销毁的当前的 Fragment 的,然后第一个 Fragment 只是从不可见的状态变为了可见的状态,并没有走相关的生命周期,所以

hide() 
的方法不会触发
onPause() 
等生命周期回调方法。

那么如果我们锁屏了或者切换任务之后再切换回来的话:


E/TAG: onPause: Fragment09
 E/TAG: onPause: Fragment10
 E/TAG: onStop: Fragment09
 E/TAG: onStop: Fragment10
 E/TAG: onStart: Fragment09
 E/TAG: onStart: Fragment10
 E/TAG: onResume: Fragment09
 E/TAG: onResume:不可见的 Fragment09
 E/TAG: onResume: Fragment10
 E/TAG: onResume:可见的 Fragment10

这里可以看到,所有的 Fragment 都回去随着 Activity 去回调相关方法,不管它是否可见。

replaceTo


FragmentTransaction transaction = manager.beginTransaction();
 String tag = to.getClass().getSimpleName();
 transaction.replace(from.getContainerId(), to, tag)
   .addToBackStack(tag)
   .commit();

replace() 
的方法其实就是相当于
remove()
移除之前添加到这个容器中的所有 Fragment 然后再
add() 
添加当前的。那么既然会调用
remove() 
方法,所以生命周期就是这样的啦:


E/TAG: onAttach: Fragment09
 E/TAG: onCreate: Fragment09
 E/TAG: onCreateView: Fragment09
 E/TAG: onStart: Fragment09
 E/TAG: onResume: Fragment09
 E/TAG: onResume:可见的 Fragment09
 E/____TAG____: onClick: 2131558527
 E/TAG: onAttach: Fragment10
 E/TAG: onCreate: Fragment10
 E/TAG: onPause: Fragment09
 E/TAG: onStop: Fragment09
 E/TAG: onDestroyView: Fragment09
 E/TAG: onCreateView: Fragment10
 E/TAG: onStart: Fragment10
 E/TAG: onResume: Fragment10
 E/TAG: onResume:可见的 Fragment10

对比上面的log来看的话,当

remove() 
调用之后, Fragment 会执行
onPause() 
onStop()
onDestroyView() 
会一次被调用,但是
onDestroy() 
onDetach()
是不会被调用的,这就是说它的视图会被摧毁了,那么重新回来的时候,就得重新创建:


E/TAG: onPause: Fragment10
 E/TAG: onStop: Fragment10
 E/TAG: onDestroyView: Fragment10
 E/TAG: onDestroy: Fragment10
 E/TAG: onDetach: Fragment10
 E/TAG: onCreateView: Fragment09
 E/TAG: onStart: Fragment09
 E/TAG: onResume:可见的 Fragment09

同样的,锁屏了或者切换任务之后再切换回来的话:


E/TAG: onStart: Fragment10
 E/TAG: onResume:可见的 Fragment10
 E/TAG: onPause: Fragment10
 E/MainActivity: onSaveInstanceState: 保存当前TAG
 E/TAG: onStop: Fragment10
 E/TAG: onStart: Fragment10
 E/TAG: onResume:可见的 Fragment10

可以看到,当使用了

replace() 
之后,这些情况之下只会有top的 Fragment 来响应对应的生命周期。在上面的
add() 
中,两个 Fragment 都走了相关的生命周期的。

那么问题来了:什么时候使用

replace() 
什么时候使用
add() 
hide()
的呢?!其实对比来看,主要就是效率的问题和相关生命周期问题。

效率问题:

如果使用

replace() 
就意味着每次的需要走
onCreateView() 
再次去重新填充布局。如果在
onCreateView()
方法中还包含了初始化数据的话,也意味着相关的也要重新执行一次。

数据、页面刷新问题:

如果你使用

replace() 
从 AFragment 跳转到 BFragment , BFragment 中更新了相关数据会影响到 AFragment 的相关 View 展示的话,这里也会有问题,就算你使用 EventBus 什么的通知了, AFragment 的确可以改变,但是当你切回到 AFragment ,它会走
onCreateView()
重新创建相关布局,除非你保存到了全局,初始化的时候再次设置,那么之前发送的数据就会丢失了。

生命周期匹配问题:

生命周期的方法都是匹配成对出现的,上面说到的

replace()
方法中,A替换的时候走了以下三个生命周期回调方法:


E/TAG: onPause: Fragment09
 E/TAG: onStop: Fragment09
 E/TAG: onDestroyView: Fragment09

当回退到它的时候,对应的三个生命周期回调就被调用了:


E/TAG: onCreateView: Fragment09
 E/TAG: onStart: Fragment09
 E/TAG: onResume:可见的 Fragment09

但是使用

add() 
hide() 
的时候就会比较尴尬,你会发现它的
onPause()
onResume()
方法完全不匹配了。只要
add()
了,即使你调用
hide()
,不会影响它的生命周期回调,也不会有
onPause()
等回调。这也就出现了当我们锁屏或者切换任务回来的话,所有add进来的 Fragment 都会执行一遍相关生命周期回调方法:


E/TAG: onPause: Fragment09
 E/TAG: onPause: Fragment10
 E/TAG: onStop: Fragment09
 E/TAG: onStop: Fragment10
 E/TAG: onStart: Fragment09
 E/TAG: onStart: Fragment10
 E/TAG: onResume: Fragment09
 E/TAG: onResume:不可见的 Fragment09
 E/TAG: onResume: Fragment10
 E/TAG: onResume:可见的 Fragment10

所以,如果你做统计相关的,这里可能就有点儿小问题了。当然,它显示与否并不是完全不可知的。

在 Fragment 中可以通过

isHide()
的方法,或者
onHiddenChanged(boolean hiden) 
的方法来获取当前是否是 hide 状态。

所以总结起来就是使用

add()
hide() 
的方式,需要注意
onResume() 
等回调方法的不匹配情况和获取数据的时机,应该在可见的时候( isHidden() 返回 false 的时候)才去请求相关数据。

使用

replace() 
的话就是要注意频繁的布局填充还有就是 Fragment 与 Fragment 之间的数据传递情况。

状态保存

Fragment重叠异常情况

肯定碰到过Fragment重叠显示的问题吧!?

 

 

这个主要就是 Activity 帮我们在作相关的恢复状态的时候出现的问题。在设置中将不保留后台进程打开,方便产生对应的情况:

然后在第二个页面时回到首页,再次进入,我们先来看看一个异常的情况的log:


E/TAG: onAttach: Fragment09
 E/TAG: onCreate: Fragment09
 E/TAG: onAttach: Fragment10
 E/TAG: onCreate: Fragment10
 E/TAG: onCreateView: Fragment09
 E/TAG: onCreateView: Fragment10
 E/TAG: onAttach: Fragment09
 E/TAG: onCreate: Fragment09
 E/TAG: onCreateView: Fragment09
 E/TAG: onStart: Fragment09
 E/TAG: onStart: Fragment10
 E/TAG: onStart: Fragment09
 E/TAG: onResume:不可见的 Fragment09
 E/TAG: onResume:可见的 Fragment10
 E/TAG: onResume:可见的 Fragment09

尼玛,发现问题没?我们的 Fragment9 居然创建了两次。一个和之前是一样的,不可见的,另外一个居然是可见的,而且还在最上面。所以这个就造成了 Fragment 重叠的情况。

还有一种情况就是断点发现某个 Fragment 初始化成功了,布局也有了,但是里面的View全是空的,这个情况我也遇到过。

为什么会出现这个问题?因为在这种异常情况下,会触发 Android 的临时数据保存机制, Fragment 是它临时保存的重点对象。所以之前的两个 Fragment 相关状态都被保存下来了!但在 Activity 的

onCreate() 
中我是这样写的话:


fragmentsUtil.loadRoot(R.id.fragment_container, Fragment9.newInstance());

那么就是说不管是否有保存的状态我都去再次创建加载一次 Fragment9 了,所以这个就导致了 Fragment9 创建了两次(一次是系统恢复出来的,相关状态也是正常的,一次就是我们在

onCreate()
中创建出来的)。那么要避免这个问题就要在 savedInstanceState 这个东西上想想办法了。既然它已经保存了相关的 Fragment 了,我们就不用去再次创建咯!


@Override
protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_main);
 ButterKnife.bind(this);
 manager = getSupportFragmentManager();
 if (savedInstanceState == null) {
  Log.e(TAG, "onSaveInstanceState: 恢复相关状态!!");
  fragmentsUtil.loadRoot(R.id.fragment_container, Fragment9.newInstance());
 }
}

所以说呢, savedInstanceState 还是不能忽略的。不过到这里就以为完了的话就太低估 Fragment 的坑了, 如果你的 support-library 是低于24的,那么即使判断了 savedInstanceState 再去创建但是也有可能出现重叠的情况!

这里要说说 FragmentState 这个类,它是用来保存 Fragment 的相关状态的。

发现情况没??在之前的 FragmentState 中并没有保存 mHidden 的状态!

另外也搜索到了 相关Issue提交 ,但是在 Android 的 Revision-History 中并没有看到提及的相关bug修复。所以具体什么时候正式加了 mHidden 的字段也就没有考证出来了。

聊了这么多,最后说说解决方案呗,其实原理很简单,既然它没有自动保存,那么我们就在保存状态的时候手动把 mHidden 状态保存,在初始化的时候根据保存的 mHidden 状态手动显示或者隐藏。


@Override
public void onSaveInstanceState(Bundle outState) {
 //手动保存
 outState.putBoolean(ARG_IS_HIDDEN, isHidden());
 super.onSaveInstanceState(outState);
}
 //onCreate的时候调用
public void initFragments(Bundle savedInstanceState, BaseFragment fragment) {
 if (savedInstanceState == null) {
  return;
 }
 boolean isSupportHidden = savedInstanceState.getBoolean(ARG_IS_HIDDEN);
 FragmentTransaction ft = manager.beginTransaction();
 if (isSupportHidden) {
  ft.hide(fragment);
 } else {
  ft.show(fragment);
 }
 ft.commit();
}

最后封装了一个简单的工具类和 BaseFragment , 用于处理 Fragment 相关的事务的。

总结

以上就是这篇文章的全部内容了,希望本文的内容对各位Android开发者们能有所帮助,如果有疑问大家可以留言交流。

您可能感兴趣的文章:Android design包自定义tablayout的底部导航栏的实现方法Android 中TabLayout自定义选择背景滑块的实例代码android TabLayout使用方法详解Android TabLayout(选项卡布局)简单用法实例分析Android 开发中fragment预加载问题Android中ViewPager获取当前显示的FragmentAndroid中fragment与activity之间的交互(两种实现方式)Android实现Tab布局的4种方式(Fragment+TabPageIndicator+ViewPager)Android使用TabLayout+Fragment实现顶部选项卡


免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Android中Fragment管理及重叠问题的解决方法

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android中Fragment管理及重叠问题的解决方法

一、Fragment介绍fragment在3.0被引入以后,项目使用fragment越来越多,特别是主界面是底部tab页点击切换更换内容,当然啦, Fragment 在项目中存在着广泛的时候,例如通常在首页的设计中,通常底部的每一个 nav
2022-06-06

Android中Fragment 重叠遮盖问题解决办法

1.导致Fragment 重叠 和遮盖的原因 主要还是因为Fragment的状态保存机制,当系统内存不足时,Fragment的主Activity被回收,Fragment的实例并没有随之被回收。Activity被系统回收时,会主动调用onSa
2023-05-31

Android中fragment嵌套fragment问题解决方法

都说fragment好用,duang~~,又遇到问题了,记录一下,分享给遇到这个问题的同学! 1.fragment嵌套fragment时出现getActivity()为null activity A嵌套fragment B,
2022-06-06

AndroidBottomNavigationView与Fragment重建与重叠问题解决方法探索

这篇文章主要介绍了AndroidBottomNavigationView与Fragment重建与重叠问题解决,总的来说这并不是一道难题,那为什么要拿出这道题介绍?拿出这道题真正想要传达的是解题的思路,以及不断优化探寻最优解的过程。希望通过这道题能给你带来一种解题优化的思路
2023-01-17

Android Fragment多层嵌套重影问题的解决方法

1解决bug的思想: //step1:当bug被发现(排除极低偶然性,单次性,开发工具导致) //step2:根据经验判断bug的重现场景,多次测试,直到精准的定位bug //step3:根据重现场景找到对应的代码 //step4:分析区域
2022-06-06

C#中常见的内存管理问题及解决方法

C#中常见的内存管理问题及解决方法,需要具体代码示例在C#开发中,内存管理是一个重要的问题,不正确的内存管理可能会导致内存泄漏和性能问题。本文将向读者介绍C#中常见的内存管理问题,并提供解决方法,并给出具体的代码示例。希望能帮助读者更好地理
2023-10-22

Android应用中使用Fragment组件的一些问题及解决方案总结

Fragment的主要意义就是提供与Activity绑定的生命周期回调。 Fragment不一定要向Activity的视图层级中添加View. 当某个模块需要获得Activity的生命周期回调的时候,就可以考虑通过Fragment来实现.
2022-06-06

C++中的函数重载问题及解决方法

C++中的函数重载问题及解决方法引言:函数重载是C++中一种非常强大的特性,它允许在同一个作用域内定义多个同名函数,但函数的参数类型、个数或顺序不同。这样可以根据不同的参数选择不同的函数执行,提高代码的灵活性和可读性。然而,在实际编程过程中
2023-10-22

C++技术开发中的内存管理问题及解决方法

C++技术开发中的内存管理问题及解决方法在C++开发中,内存管理是一个关键的问题。不正确的内存管理可能导致内存泄漏、野指针访问、内存溢出等严重的后果。本文将探讨一些常见的内存管理问题,并提供相应的解决方法和示例代码。内存泄漏内存泄漏指的是程
2023-10-22

Android中Fragment多层嵌套时onActivityResult无法正确回调问题的解决方法

前言:Fragment也可以使用startActivityForResult方法去打开一个Activity,然后在其onActivityResult方法中处理结果,可是当Fragment嵌套的时候,由于FragmentActivity的BU
2022-06-06

Android开发中怎么解决Fragment +Viewpager滑动页面重复加载的问题

这篇文章给大家分享的是有关Android开发中怎么解决Fragment +Viewpager滑动页面重复加载的问题的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。前言 之前在做一个Viewpager上面加载多个Fr
2023-05-30

C++中函数重载问题及解决方法的概述

C++中函数重载问题及解决方法的概述在C++中,函数重载是指在同一个作用域中可以定义多个同名但参数类型或参数个数不同的函数。函数重载的好处在于能够提高代码的可读性和灵活性,使得开发人员可以根据不同的需求使用同一个函数名进行操作。然而,函数重
2023-10-22

C++中多重继承问题及解决方法的介绍

C++中多重继承问题及解决方法的介绍在C++中,多重继承是一种强大的特性,允许一个类从多个父类派生而来。然而,多重继承也带来了一些问题和挑战,其中最常见的问题是菱形继承问题(Diamond Inheritance Problem)。菱形继承
2023-10-22

C++中函数重载问题及解决方法的介绍

C++中函数重载问题及解决方法的介绍在C++中,函数重载是指在同一个作用域内,使用相同的函数名,但函数参数的类型、个数或顺序不同的情况下,定义多个函数的一种机制。通过函数重载,我们可以为相同的操作或功能提供不同的实现方式,以便满足不同的需求
2023-10-22

C++中多重继承问题及解决方法概述

C++中多重继承问题及解决方法概述引言:在面向对象编程中,继承是一种重要的代码复用机制。C++支持多重继承,即一个子类可以同时从多个父类继承属性和方法。然而,多重继承也带来了一些问题,如命名冲突和二义性。本文将讨论多重继承问题,并介绍解决方
2023-10-22

Android中ImageView无法居中的问题解决方法

代码如下:[java] 代码如下:
2022-06-06

C++中指针问题及引用问题的解决方法

C++中指针问题及引用问题的解决方法在C++编程中,指针是一种非常重要的数据类型,允许我们直接访问内存地址。然而,指针也经常会导致一些问题,例如空指针引用和悬空指针引用。此外,我们还经常会遇到引用问题,例如引用类型的函数参数传递和返回值引用
2023-10-22

Python开发中遇到的内存管理问题及解决方案

Python开发中遇到的内存管理问题及解决方案摘要:在Python开发过程中,内存管理是一个重要的问题。本文将讨论一些常见的内存管理问题,并介绍相应的解决方案,包括引用计数、垃圾回收机制、内存分配、内存泄漏等。并提供了具体的代码示例来帮助读
2023-10-22

Android Studio无法执行Java类的main方法问题及解决方法

Android Studio升级到哦最新版3.6.1后,新建了个项目,发现无法执行Java类的main方法。试了网上的各种方法,比如切换gradle离线模式、gradle.properties中添加android.enableAapt2=f
2022-06-06

Python中异常处理的常见问题及解决方法

Python中异常处理的常见问题及解决方法引言:在编写程序时,很难避免出现各种各样的错误和异常。异常处理是一种机制,可以在程序运行时捕获和处理这些异常,从而保证程序的稳定性和可靠性。在Python中,异常处理是一项非常重要的技能,本文将介绍
2023-10-22

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录