iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >SharedPreference引发ANR原理是什么
  • 907
分享到

SharedPreference引发ANR原理是什么

2023-07-05 05:07:32 907人浏览 泡泡鱼
摘要

这篇文章主要介绍“SharedPreference引发ANR原理是什么”,在日常操作中,相信很多人在SharedPreference引发ANR原理是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Share

这篇文章主要介绍“SharedPreference引发ANR原理是什么”,在日常操作中,相信很多人在SharedPreference引发ANR原理是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”SharedPreference引发ANR原理是什么”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

正文

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

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

// 初始化一个spSharedPreferences sharedPreferences = context.getSharedPreferences("name_sp", MODE_PRIVATE);// 修改key的值,有两种方法:commit和applysharedPreferences.edit().putBoolean("key_test", true).commit();sharedPreferences.edit().putBoolean("key_test", true).apply();// 读取一个keysharedPreferences.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原理是什么”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

--结束END--

本文标题: SharedPreference引发ANR原理是什么

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

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

本篇文章演示代码以及资料文档资料下载

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

下载Word文档
猜你喜欢
  • SharedPreference引发ANR原理是什么
    这篇文章主要介绍“SharedPreference引发ANR原理是什么”,在日常操作中,相信很多人在SharedPreference引发ANR原理是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Share...
    99+
    2023-07-05
  • SharedPreference引发ANR原理详解
    目录正文SharedPreference问题总结正文 日常开发中,使用过SharedPreference的同学,肯定在监控平台上看到过和SharedPreference相关的ANR...
    99+
    2023-02-21
    SharedPreference引发ANR SharedPreference ANR
  • Android ANR的原理是什么
    本篇内容介绍了“Android ANR的原理是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一、ANR说明和原因1.1 简介A...
    99+
    2023-06-21
  • 什么是MySQL索引原理
    下面一起来了解下什么是MySQL索引原理,相信大家看完肯定会受益匪浅,文字在精不在多,希望什么是MySQL索引原理这篇短内容是你想要的。 索引原理&本质MySQL官方解释:索引是为MySQ...
    99+
    2022-10-18
  • MySQL索引原理是什么
    这篇文章主要介绍MySQL索引原理是什么,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!摘要: 就一起来聊一聊MySQL索引。 什么是索引? 百度百科是这样描述的: 索引是为来加速对表...
    99+
    2022-10-18
  • 什么是mysql的索引原理
    下面讲讲关于mysql的索引原理,文字的奥妙在于贴近主题相关。所以,闲话就不谈了,我们直接看下文吧,相信看完mysql的索引原理这篇文章你一定会有所受益。      ...
    99+
    2022-10-18
  • MySQL的索引原理是什么
    本篇内容介绍了“MySQL的索引原理是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1、索引的本质索引...
    99+
    2022-10-18
  • MySQL索引的原理是什么
    本篇内容介绍了“MySQL索引的原理是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!索引,可能让好很多...
    99+
    2022-10-18
  • Lucene倒排索引原理是什么
    本篇内容主要讲解“Lucene倒排索引原理是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Lucene倒排索引原理是什么”吧!一、搜索引擎介绍1.1 搜索引擎是什么这里引用百度百科的介绍:搜...
    99+
    2023-06-02
  • mysql中innodb索引原理是什么
    这篇文章主要介绍了mysql中innodb索引原理是什么,具有一定借鉴价值,需要的朋友可以参考下。希望大家阅读完这篇文章后大有收获。下面让小编带着大家一起了解一下。聚集索引(clustered index)...
    99+
    2022-10-18
  • SqlServer中索引的原理是什么
    本篇文章为大家展示了SqlServer中索引的原理是什么,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。索引的概念索引的用途:我们对数据查询及处理速度已成为衡量应用系...
    99+
    2022-10-18
  • MySQL中索引的原理是什么
    这期内容当中小编将会给大家带来有关MySQL中索引的原理是什么,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。索引目的索引的目的在于提高查询效率,可以类比字典,如果要查“m...
    99+
    2022-10-18
  • weakMap为什么是弱引用原理
    目录前言弱引用 vs 强引用Map vs WeakMap前言 聊一下WeakMap弱引用,看了很多文章发现说的不对,给出的demo经过尝试也是错误的,哎,难搞也要搞 弱引用 vs ...
    99+
    2023-02-10
    weakMap 弱引用 weakMap 弱引用原理
  • mysql联合索引的原理是什么
    MySQL联合索引的原理是通过将多个列组合成一个索引,提高多列查询的效率。联合索引的原理可以通过B+树来解释。B+树是一种多叉树,它...
    99+
    2023-09-21
    mysql
  • innodb联合索引的原理是什么
    InnoDB联合索引的原理是将多个列的值组合起来作为索引的键,并按照指定的顺序进行排序和存储。它使用B+树数据结构来实现索引存储和查...
    99+
    2023-10-27
    innodb
  • Mysql的B+Tree索引原理是什么?
    首先,正确的创建合适的索引,是提升数据库查询性能的基础。索引是什么?索引是为了加速对表中数据行的检索而创建的一种分散存储的数据结构。索引的工作机制是怎样的?如上图中,如果现在有一条sql语句 selec&#...
    99+
    2022-10-18
  • 索引失效底层原理是什么
    这篇文章主要讲解了“索引失效底层原理是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“索引失效底层原理是什么”吧!单值索引B+树图单值索引在B+树的结构里...
    99+
    2022-10-18
  • MySQL索引的底层原理是什么
    这篇文章将为大家详细讲解有关MySQL索引的底层原理是什么,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。索引类型从索引的实现上,我们可以将其分为聚集索引与非...
    99+
    2022-10-18
  • JavaScript 引擎的运行原理是什么
    这篇文章将为大家详细讲解有关JavaScript 引擎的运行原理是什么,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。注意:本文主要基于 Node.js 和基...
    99+
    2022-10-19
  • mysql索引的工作原理是什么
    本篇内容介绍了“mysql索引的工作原理是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!  索引的工作...
    99+
    2022-10-18
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作