Android中Fragment数据保存和恢复

读完本文大概需要 4 分钟。

写在前面

上周我们总结了Activity中数据的保存和恢复,我们花两分钟来回顾一下:

Android中突发情况数据的保存和恢复

一句话总结

  • 临时数据
    对于临时数据,我们使用onSaveInstanceState方法进行保存,并且在onCreate方法中恢复。

  • 永久数据
    对于持久性数据,我们要在onPause方法中进行存储,但是要注意,onPause方法中不能进行大量操作,会影响其他Activity进入任务栈栈顶。

ps:在Activity中弹出一个当前Activity的Dialog并不会有任何生命周期方法调用(以前我曾以为会调用onPause方法)。因为Dialog作为一个View本身就是属于当前Activity的,Activity并没有失去焦点。

ok,完成了回顾,下面来开始本篇博客:

Fragment在我们的项目中真的太实用和常见了,它的使用频率和数量甚至超过了Activity,所以本文目的是探究Fragment的数据保存和恢复。

在开始讲解之前,你应该对Fragment的生命周期方法有一定了解,推荐给大家一篇博客,我认为不错:

「Fragment生命周期方法详解」

准备工作做了这么多,下面我们正式开始吧!

Android中Fragment数据保存和恢复_第1张图片

本文直接选用了「第一行代码」中Fragment模块的讲解例子,点击下面的按钮分别跳转这四个Fragment。为了方便观察,我重写了Fragment所有生命周期方法和onSaveInstanceState方法,并打印了Log。

我们目的是探究Fragment数据的保存和恢复,在这里我把它分为两大类的情况:

  • 1. 单个Fragment遭遇一些突发情况

  • 2. Fragment之间相互的切换或覆盖

在此之前,先引入一个返回栈的概念。

我想你应该知道返回栈是什么,并且你以前接触的应该是保存Activity的返回栈,类比Activity,Fragment返回栈其实是保存Fragment的栈结构。区别在于:Fragment的返回栈由Activity管理;而Activity的返回栈由系统管理。

在未修改之前,本文添加并切换Fragment的方式都是在返回栈中仅有一个 fragment:

Android中Fragment数据保存和恢复_第2张图片

不要心急,过一会再说怎么去在返回栈中压入多个fragment,我们先来处理只有一个的情况

1. 单个Fragment遭遇突发情况

仍然是用以下突发情况进行测试:

  • 点击back键

  • 点击锁屏键

  • 点击home键

  • 其他APP进入前台

  • 启动了另一个Activity

  • 屏幕方向旋转

  • APP被Kill

不过与上篇博客不同的是,我们在清单文件中,给Activity做了如下配置:

Android中Fragment数据保存和恢复_第3张图片

这么做的目的是当屏幕方向发生改变的时候,fragment所依附的Activity并不会重新销毁再创建,让情况相对简单一点。

测试结果

当一个fragment孤零零地呆在返回栈时,它所处的情况与Activity如出一辙。类比Activity对数据的保存和恢复,我们可以对此得出结论:

  • 临时数据 对于临时数据,我们使用onSaveInstanceState方法进行保存,并且在onCreateView方法中恢复(请注意是onCreateView)。

  • 永久数据 对于持久性数据,我们要在onPause方法中进行存储。

2. Fragment之间的相互切换或覆盖

当返回栈中保证只有一个Fragment,相互切换时,生命周期方法的调用是怎样的呢?例如本例中,从fragment03切换到fragment04:

Android中Fragment数据保存和恢复_第4张图片

可以看到,上述的这种情况,两个fragment从创建到销毁,经历了所有的生命周期方法。

如果返回栈中fragment的数量为多个呢?首先在切换时,加上以下代码,保证将fragment放入返回栈中:

Android中Fragment数据保存和恢复_第5张图片

使用addToBackStack方法,就能将fragment放入相应的返回栈中去了,从表象上来看区别在于进入其他fragment时,点击back键时,可以返回上一个fragment。这时候切换时,生命周期方法就是如何调用的呢?

Android中Fragment数据保存和恢复_第6张图片

对比这两张生命周期方法的图,能得出两个结论。

  • 1. 无论任务栈中fragment数量为多少,onSaveInstanceState方法都没有调用

  • 2. 当fragment任务栈中有多个fragment时,进入下一个fragment时,并不会销毁fragment实例,而是仅仅销毁视图,最终调用的方法为onDestoryView。
      

所以此时我们要去保存临时数据,并不能仅保存在onSaveInstanceState中(因为它可能不会调用),还应该在onDestoryView方法中进行保存临时数据的操作,源码如下:

Android中Fragment数据保存和恢复_第7张图片

因为没有了系统提供的bundle参数,我们选择把数据保存在Arguments中,代码就不带着大家一步一步的看了,因为逻辑并不复杂,挺好理解的。通过这种方式,我们就挺容易的将临时数据和fragment的一些状态保存进bundle中并在需要时恢复了。

不知不觉本篇文章就要结束了,感兴趣的可以尝试当调用ft.add()方式去添加fragment时,生命周期方法又是怎样调用的呢?

一句话总结本文

Fragment对临时数据的保存,仅仅依靠onSaveInstanceState方法是不行的,还需要在onDestoryView中进行相应操作,具体参考上面的代码。

Fragment中对于一些持久性的数据,仍应在onPause中保存。

投稿作者:MeloDev

原文:http://www.jianshu.com/p/015c79bedb41

你可能感兴趣的:(Android中Fragment数据保存和恢复)