返回顶部
首页 > 资讯 > 移动开发 >SharedPreference引发ANR原理详解
  • 213
分享到

SharedPreference引发ANR原理详解

SharedPreference引发ANRSharedPreferenceANR 2023-02-21 18:02:30 213人浏览 薄情痞子
摘要

目录正文SharedPreference问题总结正文 日常开发中,使用过SharedPreference的同学,肯定在监控平台上看到过和SharedPreference相关的ANR

正文

日常开发中,使用过SharedPreference的同学,肯定在监控平台上看到过和SharedPreference相关的ANR,而且量应该不小。如果使用比较多或者经常用sp存一些大数据,如JSON等,相关的ANR经常能排到前10。下面就从源码的角度来看看,为什么SharedPreference容易产生ANR。

SharedPreference的用法,相信做过Android开发的同学都会,所以这里就只简单介绍一下,不详细介绍了。

// 初始化一个sp
SharedPreferences sharedPreferences = context.getSharedPreferences("name_sp", MODE_PRIVATE);
// 修改key的值,有两种方法:commit和apply
sharedPreferences.edit().putBoolean("key_test", true).commit();
sharedPreferences.edit().putBoolean("key_test", true).apply();
// 读取一个key
sharedPreferences.getBoolean("key_test", false);

SharedPreference问题

SharedPreference的相关方法,除了commit外,一般的开发同学都会直接在主线程调用,认为这样不耗时。但其实,SharedPreference的很多方法都是耗时的,直接在主线程调很可能会引起ANR的问题。另外,虽然apply方法的调用不耗时,但是会引起生命周期相关的ANR问题。

下面就来从源码的角度,看一下可能引起ANR的问题所在。

getSharedPreference(String name, int mode)

    @Override
    public SharedPreferences getSharedPreferences(String name, int mode) {
        File file;
        //  与sp相关的操作,都使用ContextImpl的类
        synchronized (ContextImpl.class) {
            if (mSharedPrefsPaths == null) {
                mSharedPrefsPaths = new ArrayMap<>();
            }
            // mSharedPrefsPaths是内存缓存的文件路径
            file = mSharedPrefsPaths.get(name);
            if (file == null) {
                // 此处获取SharedPreferences的文件路径,可能存在耗时
                file = getSharedPreferencesPath(name);
                mSharedPrefsPaths.put(name, file);
            }
        }
        return getSharedPreferences(file, mode);
    }

下面看下获取文件路径的方法:getSharedPreferencesPath(),这个方法可能存在耗时。

    public File getSharedPreferencesPath(String name) {
        // 创建一个sp的存储文件
        return makeFilename(getPreferencesDir(), name + ".xml");
    }

调用getPreferencesDir()获取sharedPrefs的根路径

    private File getPreferencesDir() {
        // 所有和文件有关的操作,都会使用mSync锁,可能出现与其他线程抢锁的耗时
        synchronized (mSync) {
            if (mPreferencesDir == null) {
                mPreferencesDir = new File(getDataDir(), "shared_prefs");
            }
            // 这个方法,如果目录不存在,会创建目录,可能存在耗时
            return ensurePrivateDirExists(mPreferencesDir);
        }
    }

ensurePrivateDirExists():确保文件目录存在

    private static File ensurePrivateDirExists(File file, int mode, int gid, String xattr) {
        if (!file.exists()) {
            final String path = file.getAbsolutePath();
            try {
                // 创建文件夹,会耗时
                Os.mkdir(path, mode);
                Os.chmod(path, mode);
            } catch (ErrnoException e) {
            }
        return file;
    }

再来看看getSharedPreferences生成SharedPreferenceImpl对象的流程。

    public SharedPreferences getSharedPreferences(File file, int mode) {
        SharedPreferencesImpl sp;
        synchronized (ContextImpl.class) {
            // 获取cache,先从cache中获取SharedPreferenceImpl
            final ArrayMap<File, SharedPreferencesImpl> cache = getSharedPreferencesCacheLocked();
            sp = cache.get(file);
            if (sp == null) {
                // 如果没有cache,则创建一个SharedPreferencesImpl,此处可能存在耗时
                sp = new SharedPreferencesImpl(file, mode);
                cache.put(file, sp);
                return sp;
            }
        }
        return sp;
    }

先来看下cache的原理

    private ArrayMap<File, SharedPreferencesImpl> getSharedPreferencesCacheLocked() {
        // ssharedPrefsCache是一个静态变量,全局有效
        if (sSharedPrefsCache == null) {
            sSharedPrefsCache = new ArrayMap<>();
        }
        // key:包名,value: ArrayMap<File, SharedPreferencesImpl> 
        final String packageName = getPackageName();
        ArrayMap<File, SharedPreferencesImpl> packagePrefs = sSharedPrefsCache.get(packageName);
        if (packagePrefs == null) {
            packagePrefs = new ArrayMap<>();
            sSharedPrefsCache.put(packageName, packagePrefs);
        }
        return packagePrefs;
    }

再来看看SharedPreferenceImpl的构造方法,看看SharedPreference是怎么初始化的。

    SharedPreferencesImpl(File file, int mode) {
        mFile = file;
        mBackupFile = makeBackupFile(file);
        // 设置是否load到内存的标志位为false
        mLoaded = false;
        startLoadFromDisk();
    }

startLoadFromDisk():开启一个子线程,将sp中的内容读取到内存中

    private void startLoadFromDisk() {
        // 改mLoaded标志位时,需要获取mLock锁
        synchronized (mLock) {
           // load之前先设置mLoaded标志位为false
            mLoaded = false;
        }
        // 开启一个线程,从文件中将sp中的内容读取到内存中
        new Thread("SharedPreferencesImpl-load") {
            public void run() {
                // 在子线程load
                loadFromDisk();
            }
        }.start();
    }

loadFromDisk:真正读取文件的地方

   private void loadFromDisk() {
        synchronized (mLock) {
            // 如果已经load过了,直接return,不需要再重新load
            if (mLoaded) {
                return;
            }
            stat = Os.stat(mFile.getPath());
            if (mFile.canRead()) {
                BufferedInputStream str = null;
                try {
                    str = new BufferedInputStream(
                            new FileInputStream(mFile), 16 * 1024);
                    // 读取xml的内容到map中
                    map = (Map<String, Object>) XmlUtils.readMapXml(str);
                } catch (Exception e) {
                    Log.w(TAG, "Cannot read " + mFile.getAbsolutePath(), e);
                } finally {
                    IoUtils.closeQuietly(str);
                }
            }
        synchronized (mLock) {
            // 设置mLoaded标志位为true,表示已经load完,通知所有在等待的线程
            mLoaded = true;
            mLock.notifyAll();
        }
    }

总结:经过上面的分析,getSharedPreferences主要的卡顿点在于,获取PreferencesDir的时候,可能存在目录尚未创建的情况。如果这个时候调用了创建目录的方法,就会非常耗时。

getBoolean(String key, boolean defValue)

这个方法和所有获取key的方法一样,都可能存在耗时。

SharedPreferencesImpl的构造方法,我们知道会开启一个新的线程,将内容从文件中读取到缓存的map里,这个步骤我们叫load。

    public boolean getBoolean(String key, boolean defValue) {
        synchronized (mLock) {
            // 需要等待,直到load成功
            awaitLoadedLocked();
            // 从缓存中取value
            Boolean v = (Boolean)mMap.get(key);
            return v != null ? v : defValue;
        }
    }

主要耗时的方法,在awaitLoadedLocked里。

    private void awaitLoadedLocked() {
       // 只有当mLoaded为true时,才能跳出死循环
        while (!mLoaded) {
            try {
                // 调用wait后,会释放mLock锁,并且进入等待池,等待load完之后的唤醒
                mLock.wait();
            } catch (InterruptedException unused) {
            }
        }
    }

这个方法,调用了mLock.wait(),释放了mLock的对象锁,并且进入等待池,直到load完被唤醒。

总结:所以,getBoolean等获取key的方法,会等待,直到sp的内容从文件中copy到缓存map里。很可能存在耗时。

commit()

commit()方法,会进行同步写,一定存在耗时,不能直接在主线程调用。

        public boolean commit() {
            // 开始排队写
            SharedPreferencesImpl.this.enqueueDiskWrite(
                mcr, null );
            try {
                // 等待同步写的结果
                mcr.writtenToDiskLatch.await();
            } catch (InterruptedException e) {
                return false;
            } finally {
            }
            notifyListeners(mcr);
            return mcr.writeToDiskResult;
        }

apply()

大家都知道apply方法是异步写,但是也可能造成ANR的问题。下面我们来看apply方法的源码。

        public void apply() {
            // 先将更新写入内存缓存
            final MemoryCommitResult mcr = commitToMemory();
            // 创建一个awaitCommit的runnable,加入到QueuedWork中
            final Runnable awaitCommit = new Runnable() {
                    @Override
                    public void run() {
                        try {
                            // 等待写入完成
                            mcr.writtenToDiskLatch.await();
                        } catch (InterruptedException ignored) {
                        }
                    }
                };
            // 将awaitCommit加入到QueuedWork中
            QueuedWork.addFinisher(awaitCommit);
            Runnable postWriteRunnable = new Runnable() {
                    @Override
                    public void run() {
                        awaitCommit.run();
                        QueuedWork.removeFinisher(awaitCommit);
                    }
                };
            // 真正执行sp持久化操作,异步执行
            SharedPreferencesImpl.this.enqueueDiskWrite(mcr, postWriteRunnable);
            // 虽然还没写入文件,但是内存缓存已经更新了,而listener通常都持有相同的sharedPreference对象,所以可以使用内存缓存中的数据
            notifyListeners(mcr);
        }

可以看到这里确实是在子线程进行的写入操作,但是为什么说apply也会引起ANR呢?

因为在ActivityService的一些生命周期方法里,都会调用QueuedWork.waitToFinish()方法,这个方法会等待所有子线程写入完成,才会继续进行。主线程等子线程,很容易产生ANR问题。

public static void waitToFinish() {
       Runnable toFinish;
       //等待所有的任务执行完成
       while ((toFinish = sPendingWorkFinishers.poll()) != null) {
           toFinish.run();
       }
   }

Android 8.0 在这里做了一些优化,但还是需要等写入完成,无法完成解决ANR的问题。

总结

综上所述,SharedPreference可能在以下几种情况下产生卡顿,从而引起ANR:

  • 创建SharedPreference时,调用getPreferenceDir,可能存在创建目录的行为
  • getBoolean等方法,会等待直到SharedPreference将文件中的键值对全部读取到缓存里,才会返回
  • commit方法直接同步写,如果不小心在主线程调用,会引起卡顿
  • apply方法虽然是在异步线程写入,但是由于ActivityService的生命周期会等待所有SharedPreference的写入完成,所以可能引起卡顿和ANR问题

SharedPreference从设计之初,就是为了存储少量key-value对,而存在的。其本身的设计,就存在很多缺陷。在存储特别少量数据的时候,性能瓶颈还不显著。但是现在很多开发同学在使用的时候,会往里面存一些大型的jsON字符串等,导致它的缺点被明显暴露出来。建议在使用SharedPreference的时候,只用于存储少量数据,不要存大的字符串。

当然,我们也有一些方法来统一优化SharedPreference,减少ANR的发生,下一篇我们继续讲。

以上就是SharedPreference引发ANR原理详解的详细内容,更多关于SharedPreference引发ANR的资料请关注编程网其它相关文章!

--结束END--

本文标题: SharedPreference引发ANR原理详解

本文链接: https://lsjlt.com/news/196992.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • SharedPreference引发ANR原理详解
    目录正文SharedPreference问题总结正文 日常开发中,使用过SharedPreference的同学,肯定在监控平台上看到过和SharedPreference相关的ANR...
    99+
    2023-02-21
    SharedPreference引发ANR SharedPreference ANR
  • SharedPreference引发ANR原理是什么
    这篇文章主要介绍“SharedPreference引发ANR原理是什么”,在日常操作中,相信很多人在SharedPreference引发ANR原理是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Share...
    99+
    2023-07-05
  • MySQL索引原理详解
    目录索引是什么索引数据结构树形索引树的动画为什么不是简单的二叉树?为什么不是红黑树?为什么最终选择B+树 而不是B树水平方向可以存放更多的索引key数据量估算叶子节点包含所有的索引字段叶子节点直接包含双向指针,范围查找效...
    99+
    2022-08-19
    MySQL索引原理 MySQL索引
  • MySQL的InnoDB索引原理详解
      摘要:   本篇介绍下Mysql的InnoDB索引相关知识,从各种树到索引原理到存储的细节。   InnoDB是Mysql的默认存储引擎(Mysql5.5.5之前是MyISAM,文档)。本着高效学习的...
    99+
    2022-05-18
    mysql InnoDB
  • 深入学习Android ANR 的原理分析及解决办法
    目录一、ANR说明和原因1.1 简介1.2 原因1.3 避免二、ANR分析办法2.1 ANR重现2.2 ANR分析办法一:Log2.3 ANR分析办法二:traces.txt2...
    99+
    2022-06-07
    anr Android
  • Java并发之CAS原理详解
    目录开端1.代码1.1修改后的代码1.2代码改进:CAS模仿2.CAS分析2.1Java对CAS的支持2.2CAS实现原理是什么?2.3CAS存在的问题2.3.1什么是ABA问题?2...
    99+
    2024-04-02
  • 详解vue-amap引入高德JSAPI的原理
    目录vue-amap使用vue-amap入口文件initAMapApiLoader方法AMapAPILoader类_getScriptSrc方法loadUIAMap方法总结vue-a...
    99+
    2024-04-02
  • Mysql执行原理之索引合并详解
    mysql执行原理之索引合并详解 我们前边说过MySQL在一般情况下执行一个查询时最多只会用到单个二级索引,但存在有特殊情况,在这些特殊情况下也可能在一个查询中使用到多个二级索引,MySQL中这种使用到多个索引来完成一次...
    99+
    2022-12-20
    Mysql索引合并 Mysql执行原理
  • Nodejs高并发原理示例详解
    目录导读什么是事件循环事件循环详解每个循环阶段内容详解走进案例解析nextTick 与 setImmediatenextTick 递归的危害setImmediat...
    99+
    2022-11-13
    Nodejs高并发原理 Nodejs 高并发
  • EventBus详解 (详解 + 原理)
    一、EventBus的使用介绍 EventBus简介 EventBus是一个开源库,由GreenRobot开发而来,是用于Android开发的 “事件发布—订阅总线”, 用来进行模块间通信、解藕。它可以使用很少的代码,来实现多组件之间...
    99+
    2023-08-31
    android
  • MySQL索引机制的详细解析及原理
    目录一.索引的类型与常见的操作二.常见的索引详解与创建三.索引的原理1.通过实验介绍B+tree2.延伸四.聚簇索引和非聚簇索引1.使用聚簇索引的优势2.什么情况下无法使用索引总结一...
    99+
    2024-04-02
  • 如何理解MySQL索引原理
    本篇内容主要讲解“如何理解MySQL索引原理”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何理解MySQL索引原理”吧!案例背景假设面试官问你:在电商平台的订...
    99+
    2024-04-02
  • MySQL的索引原理以及查询优化详解
    目录一、介绍1.什么是索引?2.为什么要有索引呢?二、索引的原理一 索引原理二 磁盘IO与预读三、索引的数据结构四、Mysql索引管理一、功能二、MySQL的索引分类三、 索引的两大...
    99+
    2024-04-02
  • JAVAsynchronized原理详解
    目录1、synchronized的作用2、synchronized的语法3、Monitor原理4、synchronized的原理4.1偏向锁4.2轻量级锁4.3锁膨胀4.4重量级锁4...
    99+
    2024-04-02
  • React 原理详解
    目录1.setState() 说明1.1 更新数据1.2 推荐语法1.3 第二个参数2.JSX 语法的转化过程3.组件更新机制4.组件性能优化4.1 减轻 state4.2 避免不必...
    99+
    2024-04-02
  • Nacos 原理详解
    一. 背景 现如今市面上注册中心的轮子很多,我实际使用过的就有三款:Eureka、Nacos,Zookeeper、Consul 由于当前参与Nacos 集群的维护和开发工作,期间也参与了 Nacos ...
    99+
    2023-09-03
    服务发现 java 微服务 nacos nacos源码
  • Python中弱引用的神奇用法与原理详解
    目录背景典型用法工作原理实现细节总结背景 开始讨论弱引用( weakref )之前,我们先来看看什么是弱引用?它到底有什么作用? 假设我们有一个多线程程序,并发处...
    99+
    2024-04-02
  • Java ShutdownHook原理详解
    目录ShutdownHook介绍ShutdownHook原理ShutdownHook的数据结构与执行顺序ShutdownHook触发点Shutdown.exitShutdown.sh...
    99+
    2024-04-02
  • mysql mvcc 原理详解
    前言 很多人在谈起mysql事务的时候都能很快的答出mysql的几种事务隔离级别,以及在各自隔离级别下产生的问题,但是一旦谈到为什么会产生这样的结果时会觉得难以回答,说到底,还是对底层的原理未做深入的探究,本篇将从较为底层的原理层面来聊聊...
    99+
    2023-09-17
    mvcc mysql mvcc原理 mvcc 原理详解 mysql mvcc说明 mysql mvcc详解
  • 怎样理解MySQL索引底层原理
    这篇文章给大家介绍怎样理解MySQL索引底层原理,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Mysql 作为互联网中非常热门的数据库,其底层的存储引擎和数据检索引擎的设计非常重要,尤...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作