是时候使用SaveState了

Android系统在5.0时,对进程内的内存管理做了一个优化,但并没有明确的文档说明这个优化。

这个优化为解决Android应用的内存问题,提供了一个新的思路。但如果开发者习惯于单Task的应用开发,或者从来不考虑SaveState,那开发者可能根本无法体会这个新机制的好处。

本文首先从SaveState讲起,对于了解SaveState的同学,可以直接跳过

什么是SaveState

要了解什么是SaveState必须要先知道Activity的两个关键方法

  • onSaveInstanceState
  • onRestoreInstanceState

onSaveInstanceState时系统做了些什么

Activity被回收之前,系统会调用onSaveInstanceState(Bundle outState)来保存View的状态,并到传入的outState对象中。

  1. 保存Window
  2. 保存Fragment
  3. 调用外部注册的回调方法
protected void onSaveInstanceState(Bundle outState) {
    outState.putBundle(WINDOW_HIERARCHY_TAG, mWindow.saveHierarchyState());
    Parcelable p = mFragments.saveAllState();
    if (p != null) {
        outState.putParcelable(FRAGMENTS_TAG, p);
    }
    getApplication().dispatchActivitySaveInstanceState(this, outState);
}

onRestoreInstanceState时系统做了些什么

Activity被重新创建时,会通过onCreate(Bundle savedInstanceState)onRestoreInstanceState(Bundle savedInstanceState)传入保存的状态信息并恢复View的状态。

  1. onCreate重建Fragment
  2. onRestoreInstanceState恢复Window状态
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        if (DEBUG_LIFECYCLE) Slog.v(TAG, "onCreate " + this + ": " + savedInstanceState);
        if (mLastNonConfigurationInstances != null) {
            mFragments.restoreLoaderNonConfig(mLastNonConfigurationInstances.loaders);
        }
        if (mActivityInfo.parentActivityName != null) {
            if (mActionBar == null) {
                mEnableDefaultActionBarUp = true;
            } else {
                mActionBar.setDefaultDisplayHomeAsUpEnabled(true);
            }
        }
        if (savedInstanceState != null) {
            Parcelable p = savedInstanceState.getParcelable(FRAGMENTS_TAG);
            mFragments.restoreAllState(p, mLastNonConfigurationInstances != null
                    ? mLastNonConfigurationInstances.fragments : null);
        }
        mFragments.dispatchCreate();
        getApplication().dispatchActivityCreated(this, savedInstanceState);
        if (mVoiceInteractor != null) {
            mVoiceInteractor.attachActivity(this);
        }
        mCalled = true;
    }
protected void onRestoreInstanceState(Bundle savedInstanceState) {
    if (mWindow != null) {
        Bundle windowState = savedInstanceState.getBundle(WINDOW_HIERARCHY_TAG);
        if (windowState != null) {
            mWindow.restoreHierarchyState(windowState);
        }
    }
}

Window在save和restore时对View的处理

  1. Save时,遍历View的树状结构调用 Parcelable onSaveInstanceState()
  2. 以View的id为key在Window的SparseArray<Parcelable>中保存这些 Parcelable
  3. Restore时,Window从savedInstanceState获取View的savedStates
  4. 遍历View的树状结构调用 onRestoreInstanceState(Parcelable state)
  5. View根据id获取自己的state并恢复

小结

  1. Save和Restore的机制主要是用于保存和恢复View的
  2. 没有id的View是不会被保存状态的
  3. 如果id重复,则View的状态会被覆盖
  4. 被保存的Fragment会在onCreate中被自动创建和添加到FragmentActivity中
  5. 被保存的View不会被自动创建,只是通过id获取savedInstance用于更新View

关于SaveState的详细介绍可以参考文章Android中SaveState原理分析

为什么开始使用SaveState

为什么很多人不重视SaveState

我们先了解下会用到Restore机制的地方

  1. FragmentStatePagerAdapter用于在ViewPager中使用可回收和重建的Fragment
  2. 应用Crash时,当前页面被销毁,前一个页面被Restore
  • 在4.0之前,系统不会自动重启应用
  • 在4.0之后,系统会自动重启,并通过Restore机制恢复Crash的页面。

FragmentStatePagerAdapter中考虑SaveState是必须的,所以大家都会被迫处理SaveState的问题。

大多数开发者不会考虑Crash重建的问题,所以SaveState很少被开发者重视。而认真考虑过Crash重建的开发者一定不会对SaveState陌生。

5.0的新机制

在5.0中,SaveState有了新的作用,稍加利用,它会帮你解决OutOfMemory。而根据Google的统计,到今年下半年,Android5.0及以上的系统占比将超过50%。

要触发这个新机制,你的应用必须是多Task结构的。关于Task,那又是一个很大的话题,下面我只用一个简单的例子看看这个新机制。

演示代码可以通过git仓库下载

这里看看关键的ActivivtyOne.java

  • ActivityOne是standard
  • ActivityTwo是singleInstance,所以他会在单独的新的Task中
  • AcitivyOne可以启动ActivityTwo
  • ActivityOne可以不断消耗内存
public class ActivityOne extends BaseActivity {

    boolean forceOom = false;
    List<Bitmap> memory = new ArrayList<>();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_one);
    }

    public void launchTwo(View view) {
        ActivityTwo.launch(this);
    }

    /**
     * 第一次点击使内存接近进程能获取的内存上限,再次点击触发OOM
     * @param view
     */
    public void consumeMem(View view) {
        new Thread(new Runnable() {
            @Override
            public void run() {
                while (true) {
                    if (!forceOom && isLowMemory()) {
                        forceOom = true;
                        break;
                    }
                    memory.add(Bitmap.createBitmap(1000, 1000, Bitmap.Config.ARGB_8888));
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {

                    }
                }
            }
        }).start();
    }


    /**
     * 判断已使用的内存是否接近了单进程的内存上限
     *
     * @return
     */
    public boolean isLowMemory() {
        ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
        long total = Runtime.getRuntime().totalMemory() / (1024l * 1024l);
        int max = activityManager.getMemoryClass();
        Log.w(getClass().getSimpleName(), total + "/" + max);

        if (total > activityManager.getMemoryClass() * 0.85) return true;
        return false;
    }

    public static void launch(Activity activity) {
        activity.startActivity(new Intent(activity, ActivityOne.class));
    }
}

操作步骤:

  1. ActivityOne启动ActivityTow
  2. ActivityTwo启动ActivityOne,从而切换到老的Task中,ActivityTwo不会被销毁
  3. ActivityOne不断消耗内存,直到接近进程使用内存的上限(Android系统对每个进程使用的最大内存有一个限制)

这时通过logcat你会看到5.0的不同:

  • 5.0之前:不会有什么事情发生。再次点击消耗内存,会OOM,整个进程被杀。
  • 5.0及之后:ActivityTwo#OnDestroy会被调用,这时再启动ActivityTwo,可以看到ActivityTwo#onRestoreInstanceState的调用。
W/ActivityOne: 83571877-onCreate: TaskId-7551
W/ActivityTwo: 128141518-onCreate: TaskId-7552
W/ActivityOne: 83571877-onSaveInstanceState
W/ActivityOne: 219591424-onCreate: TaskId-7551
W/ActivityTwo: 128141518-onSaveInstanceState
W/ActivityOne: 23/192
W/ActivityOne: 42/192
W/ActivityOne: 88/192
W/ActivityOne: 111/192
W/ActivityOne: 157/192

以下是5.0系统上才会出现的

W/ActivityTwo: 128141518-onDestroy
W/ActivityOne: 176/192
W/ActivityTwo: 80252517-onCreate: TaskId-7552
W/ActivityTwo: 80252517-onRestoreInstanceState
W/ActivityTwo: 80252517-onNewIntent
W/ActivityOne: 219591424-onSaveInstanceState

因此我们可以得出结论:

5.0之后,Android进程在遇到内存瓶颈时,会通过主动销毁进程中的Acitivty来释放内存。这些被销毁的Activity都属于后台Task,当被销毁的Activity需要重新出现时,会触发Restore机制

当然,这个结论又会引起很多疑问。

  1. 为什么不销毁当前Task中的后台Activity?
  2. 如果后台Task中有多个Activity是一起销毁吗?如果后台Task中的多个Activity是属于不同的进程呢?
  3. ......

关于这些问题,需要分析源码才能找到答案,但我不希望这篇文章变成对Android源码的一次分析。我将在下篇文章,继续介绍怎么处理SaveState。

当然,即使没有SaveState在5.0上带来的好处,正确处理页面的SaveState也是保证Android应用程序健壮性的一个重要部分。

怎么处理SaveState,请看下篇文章

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,907评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,987评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,298评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,586评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,633评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,488评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,275评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,176评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,619评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,819评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,932评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,655评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,265评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,871评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,994评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,095评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,884评论 2 354

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,128评论 25 707
  • 今天是个有意义的日子。 通过今天下午出去推广的事情,和晚上喝了一顿酒,聊了一会,我对人性又有了新的认识。 中午和小...
    冷剑书生阅读 334评论 0 1
  • 看这部电影之前,我的心情是忐忑的。因为它在豆瓣上的评分只有5.1。通常低于7分的电影我都是不看的,免得浪费时间。最...
    猫猫伸懒腰阅读 1,488评论 0 0
  • 聊聊TTT 随着移动互联网深入的发展和对人们无时无刻的影响,培训行业的多样性和形式也出现了多种。国家下发的红头文件...
    林奎阅读 1,037评论 0 2
  • 这TM就很尴尬了 虚弱的现实用法 喜欢主播的请点一波关注 人与人之间的信任呢? 神回复 再不给我人头我就哭了哟 没...
    f伐木累阅读 8,551评论 0 1