美文网首页Android札记总结Android开发经验谈
不要在Android的Application对象中缓存数据!

不要在Android的Application对象中缓存数据!

作者: 张明云 | 来源:发表于2014-12-26 20:41 被阅读7433次

    说明

      这是翻译老外的一篇文章,我之前有遇到过这个问题,并且看到有人在Segmentfault上问,最主要我在StackOverflow上居然没搜到累死问题,所以觉得有必要翻译过来以便后面不会再这样处理。

    前言

      在你的App中的很多地方都需要使用到数据信息,它可能是一个session token,一次费时计算的结果等等,通常为了避免Activity之间传递数据的开销,会将这些数据通过持久化来存储。

      有人建议将这些数据放在Application对象中方便所有的Activity访问,这个解决方案简单、优雅并且是......完全错误的。

      你如果你将数据缓存到Application对象中,那么有可能你的程序最终会由于一个NullPointerException异常而崩溃掉。

    一个简单的测试程序

      这是自定义Application的代码:

    // access modifiers omitted for brevity
    class MyApplication extends Application {
     
        String name;
     
        String getName() {
            return name;
        }
     
        void setName(String name) {
            this.name = name;
        }
    }
    

      在第一个Activity中,我们将用户信息存储在Application对象中:

    // access modifiers omitted for brevity
    class WhatIsYourNameActivity extends Activity {
     
        void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.writing);
     
            // Just assume that in the real app we would really ask it!
            MyApplication app = (MyApplication) getApplication();
            app.setName("Developer Phil");
            startActivity(new Intent(this, GreetLoudlyActivity.class));
     
        }
     
    }
    

      然后在第二个Activity中通过Application获取存储的用户信息:

    // access modifiers omitted for brevity
    class GreetLoudlyActivity extends Activity {
     
        TextView textview;
     
        void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
     
            setContentView(R.layout.reading);
            textview = (TextView) findViewById(R.id.message);
        }
     
        void onResume() {
            super.onResume();
     
            MyApplication app = (MyApplication) getApplication();
            textview.setText("HELLO " + app.getName().toUpperCase());
        }
    }
    

    测试步骤

    1. 打开这个APP;

    2. 在WhatIsYourNameActivity中,你按要求输入用户名并将其缓存到MyApplication这个对象中;

    3. 接着在GreetLoudlyActivity中,程序从MyApplication对象中取出用户名并显示出来;

    4. 用户按了Home按键离开了该APP;

    5. 数小时之后,系统由于内存不足(用户在体验其它APP呢,前台的任务总是优先的嘛)会在后台将你的程序杀掉;在你重新启动该APP之前一切看上去很好,但是.....;

    6. 用户重新打开了这个APP;

    7. Android会重新创建一个之前被Kill掉的MyApplication实例并恢复GreetLoudlyActivity;

    8. GreetLoudlyActivity去获取用户名时,会因为获取的为空值报NullPointerException而崩溃掉。

    为什么会这样?

      在上面这个例子中,程序之所以会崩溃掉是因为恢复之后APP的Application对象是全新的,所以缓存在Application中的用户名成员变量为空值,在程序调用String的toUpperCase()方法时由于NullPointerException而崩溃掉。

      导致这个问题的主要原因是:Application对象并不是始终在内存中的,它有可能会由于系统内存不足而被杀掉。但Android在你恢复这个应用时并不是重新开始启动这个应用,它会创建一个新的Application对象并且启动上次用户离开时的activity以造成这个app从来没有被kill掉得假象。

      我们以为可以通过Application来缓存数据,却没想到恢复APP时直接跑了B Activity而不是先启动A Activity,最终导致的结果是程序意外的崩溃掉了。

    有哪些替代方法可用呢?

      对于数据缓存问题我也没有比较好的办法,但你可以按照下面其中一种方式来处理:

    • 通过Intent在Activity之间来传递数据(但是请别传递大量数据,这有可能导致程序异常或者ANR);

    • 使用官方推荐的方法中的一种将数据持久化,存储在磁盘中;

    • 在使用数据和句柄的时候做空值检测;

    如何模拟应用程序被杀掉?

    更新:Daniel Lew指出,最简单的方法是在DDMS中点击"Stop Porcess"杀掉你的程序,在你调试程序的时候可以这样做。

      你可以通过模拟器或者一个Root过的真机来测试实际效果:

    1. 按Home按键退出你的程序;

    2. 在控制台,敲入如下命令(Windows系统下 WIN + R -> cmd -> 回车)

       # 找到该APP的进程ID
       adb shell ps
       # 找到你APP的报名
        
       # Mac/Unix: save some time by using grep:
       adb shell ps | grep your.app.package
        
       # 按照上述命令操作后,看起来是这样子的: 
       # USER      PID   PPID  VSIZE  RSS     WCHAN    PC         NAME
       # u0_a198   21997 160   827940 22064 ffffffff 00000000 S your.app.package
        
       # 通过PID将你的APP杀掉
       adb shell kill -9 21997
        
       # APP现在被杀掉啦
      
    3. 现在在桌面长按Home按键通过后台任务管理器打开你的APP,此时系统就会重新创建一个MyApplication实例了。

    总结

      不要在Application对象中缓存数据化,这有可能会导致你的程序崩掉。请使用Intent在各组件之间传递数据,抑或是将数据存储在磁盘中,然后在需要的时候取出来。

      并不仅仅只有Application对象是这样的,其它的单例或者公有静态类也有可能会由于系统内存而被杀掉,谨记。

    相关文章

      网友评论

      • 秋若然:涨姿势了😁
      • NewChar:单例 本地 实现都相当方便啊。 用一个不相关 保存数据 也不符合面向对象的吧
        sundy_5f80:@NewChar 说的对
      • NewChar:这么写不是傻**** 么。。。。
      • hewking:复现方法,试一试
      • 薅韭菜:我正准备把我的数据 保存到 Application 里边,看到这么多说不要把数据储存到Application里边搞的我有点慌了
      • 081b02196994: 问题的关键是代码逻辑不够严谨,调用app.getName()后,没有对返回的String判空就直接setText了,和是否在Application中缓存数据没关系
        阳生植物人:其实这样写也没太大问题,因为你预期的功能是,getName()不为空的。
        这样写,在开发期,反而成了一个好处,及时发现获取到的数据和你预期的不一样。
      • ae07aabec98c:本文有严重的逻辑问题。。。~~~问题的关键是,数据的load并不总是被执行,而不是因为放在了Application实现里了~~~
        阳生植物人:Application只为了抛砖引玉,为了举例说明问题,并且文章末尾总结有提到。
        并不仅仅只有Application对象是这样的,其它的单例或者公有静态类也有可能会由于系统内存而被杀掉,谨记。
      • code点点DD:我觉得如果数据初始化在application的oncreate方法,就不会出现这个情况了。之所以出现这个情况是,application的数据依赖了a activity ,所以在重新创建application的时候造成要使用的数据没有初始化。 :blush:
        阳生植物人:但是数据时在a activity用户输入的,那如果再application的onCreate中初始化呢。
      • 大姨夫斯基:嗯 我还在别的地方看到更恐怖的标题 比你的这个前面多了俩字 “千万”

        不管加不加“千万”, 都言过其实了。 事实上没有那么恐怖,只是需要在适当的时候初始化数据, 就可以了。 而且有些数据我认为放在Application才是比较好的实践,比如当前登录的用户,某些系统设置等等。
      • steveyangyun:谬论。那这样的话,磁盘缓存数据也不行,因为用户可能清空App 的数据。
      • huhu2008:好多数据都保存在磁盘也很蛋疼啊
      • 阿敏其人:赞一个
      • 7812c2da4ed6:如果是按application中每次做网络请求,然后得到的数据保存以后使用可行吗?
        b1bf1d5f72d8:@迦陵 我也想知道这个问题 我现在实在Application的onCreate方法里做数据的初始化 我不知道当我重新启动这个Application会不会报null
      • dongorigin:一个从我最初接触Android就头疼不已的问题,关键是特别的难复现,所以十分感谢提供的复现方法
      • breakingbad:赞 曾经在一台垃圾机器上遇到过这情况

      本文标题:不要在Android的Application对象中缓存数据!

      本文链接:https://www.haomeiwen.com/subject/boiatttx.html