当前位置: 首页 > article >正文

主线程没卡但ANR了?揭秘Android SharedPreferences的ANR陷阱

主线程没卡但ANR了揭秘Android SharedPreferences的ANR陷阱在Android开发中ANRApplication Not Responding问题一直是开发者头疼的难题。特别是当应用主线程看似运行正常却突然弹出ANR对话框时这种隐形ANR往往让人摸不着头脑。本文将深入剖析SharedPreferences简称SP这一常用组件背后隐藏的ANR风险机制并提供切实可行的解决方案。1. SP异步持久化机制与ANR的关联SharedPreferences作为Android提供的数据持久化方案其apply()方法常被推荐用于替代commit()以避免主线程I/O阻塞。然而正是这个看似安全的异步操作在某些场景下会成为ANR的隐形杀手。SP的写入操作分为两个阶段内存同步立即将修改更新到内存中的Map磁盘持久化通过后台线程异步写入文件关键问题出在静态广播场景下系统会等待所有SP的异步写入完成才认为广播处理结束。这个设计初衷是为了保证进程被杀前数据能持久化但却带来了意想不到的ANR风险。典型问题场景时序主线程处理广播 → 调用SP.apply() → 广播处理完成 → SP后台线程开始写入 → 系统等待写入完成 → 超时触发ANR2. 静态广播的特殊处理机制静态广播注册在AndroidManifest.xml中系统对其有特殊处理逻辑进程优先级提升处理静态广播时进程会被临时提升为前台优先级SP写入等待系统会检查是否有未完成的SP写入操作超时机制默认前台广播超时为10秒以下代码展示了静态广播中SP的危险用法public class MyReceiver extends BroadcastReceiver { Override public void onReceive(Context context, Intent intent) { // 危险操作在静态广播中频繁apply SharedPreferences sp context.getSharedPreferences(config, MODE_PRIVATE); for (int i 0; i 100; i) { sp.edit().putInt(keyi, i).apply(); // 堆积大量写入任务 } } }3. 最佳实践与解决方案3.1 SP的正确使用姿势减少apply频次批量操作优于多次小操作避免广播中大量写入静态广播中慎用apply关键数据使用commit必要时同步写入优化后的代码示例public class SafeReceiver extends BroadcastReceiver { Override public void onReceive(Context context, Intent intent) { SharedPreferences sp context.getSharedPreferences(config, MODE_PRIVATE); SharedPreferences.Editor editor sp.edit(); // 批量操作 for (int i 0; i 100; i) { editor.putInt(keyi, i); } // 根据场景选择 if (isCriticalData) { editor.commit(); // 同步写入关键数据 } else { editor.apply(); // 异步写入普通数据 } } }3.2 现代替代方案Jetpack DataStoreGoogle推荐的SP替代方案DataStore解决了这些问题特性SharedPreferencesDataStore异步API部分支持(apply)完全支持线程安全否是类型安全否是(Proto DataStore)异常处理无完善主线程安全性低高迁移到DataStore的基本步骤// 创建DataStore val dataStore context.createDataStore(name settings) // 写入数据 suspend fun saveValue(key: String, value: Int) { dataStore.edit { settings - settings[intPreferencesKey(key)] value } } // 读取数据 val exampleValueFlow: FlowInt dataStore.data .map { prefs - prefs[intPreferencesKey(example_key)] ?: 0 }3.3 检测工具与监控方案StrictMode配置可帮助早期发现问题!-- 在Application中启用严格模式 -- application android:name.MyApp android:strictModetrue /applicationJava代码配置public class MyApp extends Application { Override public void onCreate() { super.onCreate(); StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder() .detectDiskReads() .detectDiskWrites() .penaltyLog() .build()); } }ANR监控方案对比FileObserver监控traces.txt监听ActivityManager的ANR广播SIGQUIT信号捕获第三方APM工具集成4. 疑难场景分析与优化策略4.1 多进程SP的额外风险当使用MODE_MULTI_PROCESS时SP的ANR风险会加剧每次访问都会检查文件修改时间进程间同步问题可能导致重复写入建议改用ContentProvider或直接迁移到DataStore4.2 大型SP文件的优化当SP文件过大时超过100KB即使apply也可能变慢拆分策略按功能模块拆分多个SP文件冷热数据分离考虑转用SQLite清理策略定期移除过期键值避免存储大型对象4.3 兼容性处理方案对于需要支持旧版Android的项目public class SafeSPHelper { private static final boolean IS_AT_LEAST_O Build.VERSION.SDK_INT Build.VERSION_CODES.O; public static void safeApply(SharedPreferences.Editor editor) { if (IS_AT_LEAST_O) { // 8.0使用专用后台线程 editor.apply(); } else { // 旧版本使用自定义线程池 AsyncTask.THREAD_POOL_EXECUTOR.execute(() - { editor.commit(); }); } } }在实际项目中我们发现SP的ANR问题往往在用户量增长后突然爆发。一个典型案例是某社交应用在广播中频繁更新用户状态当DAU超过百万后ANR率飙升了3倍。通过将SP迁移到DataStore并结合分批写入策略最终将ANR率降低至原来的1/10。

相关文章:

主线程没卡但ANR了?揭秘Android SharedPreferences的ANR陷阱

主线程没卡但ANR了?揭秘Android SharedPreferences的ANR陷阱 在Android开发中,ANR(Application Not Responding)问题一直是开发者头疼的难题。特别是当应用主线程看似运行正常,却突然弹出ANR对话框时,这种&…...

别再乱用List了!Unity中Queue的5个高效应用场景对比

Unity中Queue的5个高效应用场景:性能对比与实战指南 在Unity开发中,数据结构的选择往往决定了游戏性能的上限。很多开发者习惯性地使用List来解决所有问题,却忽视了Queue在特定场景下的性能优势。本文将深入分析Queue的底层原理,并…...

Arduino-Pico蓝牙开发指南:经典蓝牙与BLE HID主从模式完全解析

Arduino-Pico蓝牙开发指南:经典蓝牙与BLE HID主从模式完全解析 【免费下载链接】arduino-pico Raspberry Pi Pico Arduino core, for all RP2040 and RP2350 boards 项目地址: https://gitcode.com/gh_mirrors/ar/arduino-pico 想要在Raspberry Pi Pico上实现…...

告别系统臃肿:用Win11Debloat实现Windows性能飞跃的全方位指南

告别系统臃肿:用Win11Debloat实现Windows性能飞跃的全方位指南 【免费下载链接】Win11Debloat A simple, lightweight PowerShell script that allows you to remove pre-installed apps, disable telemetry, as well as perform various other changes to declutte…...

C++模板进阶:从特化到元编程

C模板进阶详解一、模板特化全特化template <> class Vector<bool> {// 针对bool类型的特化实现 };偏特化template <typename T> class Vector<T*> {// 针对指针类型的部分特化 };二、可变参数模板template <typename... Args> void print(Args..…...

显式启用-u_printf_float和-u_scanf_float前后的代码尺寸占用实验

本文中使用的嵌入式编译器基于arm-none-eabi-gcc&#xff0c;构建文件基于Makefile。 main.c不编写任何代码&#xff0c;保证实验的其他变量统一。源文件main.c&#xff1a; 代码模式固定为Debug&#xff0c;优化等级固定为 -Og &#xff1a; syscalls.c 系统调用库函数文件参…...

DownKyi视频管理进阶指南:从新手到专家的实践路径

DownKyi视频管理进阶指南&#xff1a;从新手到专家的实践路径 【免费下载链接】downkyi 哔哩下载姬downkyi&#xff0c;哔哩哔哩网站视频下载工具&#xff0c;支持批量下载&#xff0c;支持8K、HDR、杜比视界&#xff0c;提供工具箱&#xff08;音视频提取、去水印等&#xff0…...

面试真题集(五):高级特性与系统优化

引言 本专题聚焦原子操作、动态并行、纹理内存、协作组、CUDA Graph等高级特性,以及系统级优化思维,20道真题助你冲击大厂高薪岗位。 一、选择题(6题) 1.1 关于CUDA原子操作,下列说法错误的是?(⭐⭐) A. 原子操作可以在全局内存和共享内存上执行 B. atomicAdd 是最常…...

运维工具汇总

一、远程工具列表 1. MobaXterm site: MobaXterm隧道使用_mobaxterm怎么读-CSDN博客 二、httpclient 1. small:https://zhuanlan.zhihu.com/p/701243358 2.small2: 客户端下载 | Reqable API抓包调试 API测试一站式工具 small2: https://reqable.com/zh-CN/download/ …...

3步彻底解决Windows系统卡顿问题:开源系统清理工具实战指南

3步彻底解决Windows系统卡顿问题&#xff1a;开源系统清理工具实战指南 【免费下载链接】WindowsCleaner Windows Cleaner——专治C盘爆红及各种不服&#xff01; 项目地址: https://gitcode.com/gh_mirrors/wi/WindowsCleaner 你是否也经历过这样的场景&#xff1a;工作…...

Unity路径有中文就报错?手把手教你解决Autoware高精地图插件导入的坑

Unity路径中文报错&#xff1f;Autoware高精地图插件导入全攻略 刚接触Autoware高精地图制作的新手们&#xff0c;十有八九会在第一步就栽跟头——当你兴冲冲下载好vector_map插件&#xff0c;准备在Unity中大展拳脚时&#xff0c;却发现插件死活无法正常导入。这种挫败感我太熟…...

stanford_dl_ex代码结构深度解析:从数据加载到模型评估的完整流程

stanford_dl_ex代码结构深度解析&#xff1a;从数据加载到模型评估的完整流程 【免费下载链接】stanford_dl_ex Programming exercises for the Stanford Unsupervised Feature Learning and Deep Learning Tutorial 项目地址: https://gitcode.com/gh_mirrors/st/stanford_d…...

DS18B20寄生供电模式全解析:3.3V系统下的STM32省电测温方案

DS18B20寄生供电模式全解析&#xff1a;3.3V系统下的STM32省电测温方案 在物联网设备开发中&#xff0c;低功耗设计往往决定着产品的成败。当我们需要在电池供电环境下实现长时间温度监测时&#xff0c;DS18B20传感器的寄生供电模式配合STM32的3.3V系统&#xff0c;能为我们带来…...

STM32开发避坑指南:KEIL中__use_no_semihosting报错的终极解决方案

STM32开发避坑指南&#xff1a;KEIL中__use_no_semihosting报错的终极解决方案 在嵌入式开发领域&#xff0c;STM32凭借其出色的性能和丰富的外设资源&#xff0c;成为众多开发者的首选。然而&#xff0c;在使用KEIL MDK进行开发时&#xff0c;不少开发者都会遇到一个令人头疼的…...

动画测试与调试完全手册:animation-samples项目中的自动化测试实践

动画测试与调试完全手册&#xff1a;animation-samples项目中的自动化测试实践 【免费下载链接】animation-samples Multiple samples showing the best practices in animation on Android. 项目地址: https://gitcode.com/gh_mirrors/an/animation-samples animation-…...

STM32双路直流电机PWM驱动与霍尔编码器闭环控制

1. 项目概述DCMotorDrive 是专为 RenBuggy 平台设计的双路直流电机驱动固件模块&#xff0c;其核心目标是实现对两台独立直流电机的高精度 PWM 调速控制&#xff0c;并集成霍尔传感器反馈通道&#xff0c;支持实时速度与位移闭环。该模块并非通用电机驱动芯片&#xff08;如 L2…...

Monolog Bridge 高级用法:FingersCrossed策略与HTTP状态码激活机制完全指南 [特殊字符]

Monolog Bridge 高级用法&#xff1a;FingersCrossed策略与HTTP状态码激活机制完全指南 &#x1f680; 【免费下载链接】monolog-bridge Provides integration for Monolog with various Symfony components 项目地址: https://gitcode.com/gh_mirrors/mo/monolog-bridge …...

ComfyUI-Impact-Pack完整指南:3步掌握AI图像增强的强大工具包

ComfyUI-Impact-Pack完整指南&#xff1a;3步掌握AI图像增强的强大工具包 【免费下载链接】ComfyUI-Impact-Pack Custom nodes pack for ComfyUI This custom node helps to conveniently enhance images through Detector, Detailer, Upscaler, Pipe, and more. 项目地址: h…...

Le Git Graph分支管理:动态加载和筛选分支提交的终极指南

Le Git Graph分支管理&#xff1a;动态加载和筛选分支提交的终极指南 【免费下载链接】le-git-graph Browser extension to add git graph to GitHub website. 项目地址: https://gitcode.com/gh_mirrors/le/le-git-graph Le Git Graph是一款强大的浏览器扩展&#xff0…...

MKDV4GCL-ABB嵌入式存储芯片在智能物联网设备中的关键应用解析

1. 为什么物联网设备需要专用存储芯片&#xff1f; 第一次拆解智能家居设备时&#xff0c;我发现很多厂商都在用TF卡扩展存储。但实际使用三个月后&#xff0c;问题就来了——频繁读写导致卡片损坏&#xff0c;设备不断报存储错误。这就是典型选错存储方案的后果。物联网设备对…...

一文读懂10英寸平板尺寸:从屏幕比例到实际机身尺寸

在平板电脑市场中&#xff0c;"10英寸"这一规格始终占据着黄金地位。但当我们搜索"平板10寸多大长和宽"时&#xff0c;得到的答案往往模糊不清。作为行业观察者&#xff0c;我将为您深度解析10英寸平板尺寸的行业标准、设计逻辑及选购策略&#xff0c;带您…...

Horizon 8部署避坑指南:从AD域控、SQL Server配置到Connection Server调优的12个关键检查点

Horizon 8部署实战&#xff1a;12个关键检查点与深度排错手册 在虚拟桌面基础设施&#xff08;VDI&#xff09;的部署过程中&#xff0c;每个环节都可能成为影响最终用户体验的关键因素。本文将聚焦VMware Horizon 8部署中最容易出错的12个关键节点&#xff0c;从AD域控配置到C…...

龙芯k - 走马观碑组ST驱动移植傺

正文 异步/等待解决了什么问题&#xff1f; 在传统同步I/O操作中&#xff08;如文件读取或Web API调用&#xff09;&#xff0c;调用线程会被阻塞直到操作完成。这在UI应用中会导致界面冻结&#xff0c;在服务器应用中则造成线程资源的浪费。async/await通过非阻塞的异步操作解…...

别再只用localhost了!手把手教你用路由侠把本地宝塔面板‘搬’到公网(Windows版)

突破局域网限制&#xff1a;Windows下宝塔面板安全外网访问实战指南 你是否遇到过这样的困境&#xff1f;——在本地环境调试得心应手的项目&#xff0c;当需要向异地同事演示或临时交付客户预览时&#xff0c;却因为网络隔离而束手无策。传统解决方案要么要求部署到正式服务器…...

ComfyUI InstantID:如何实现AI绘图中的精准人脸控制?

ComfyUI InstantID&#xff1a;如何实现AI绘图中的精准人脸控制&#xff1f; 【免费下载链接】ComfyUI_InstantID 项目地址: https://gitcode.com/gh_mirrors/co/ComfyUI_InstantID ComfyUI InstantID是一款专为ComfyUI设计的原生人脸特征控制插件&#xff0c;它无需依…...

Qtile社区贡献指南:从新手到核心贡献者的完整教程

Qtile社区贡献指南&#xff1a;从新手到核心贡献者的完整教程 【免费下载链接】qtile :cookie: A full-featured, hackable tiling window manager written and configured in Python (X11 Wayland) 项目地址: https://gitcode.com/gh_mirrors/qt/qtile Qtile是一个功能…...

Laravel WebSockets终极指南:本地与Redis频道管理器深度对比

Laravel WebSockets终极指南&#xff1a;本地与Redis频道管理器深度对比 【免费下载链接】laravel-websockets Websockets for Laravel. Done right. 项目地址: https://gitcode.com/gh_mirrors/la/laravel-websockets Laravel WebSockets是一款为Laravel框架打造的高效…...

终极指南:ECCV2022-RIFE在边缘设备上的快速部署实践

终极指南&#xff1a;ECCV2022-RIFE在边缘设备上的快速部署实践 【免费下载链接】ECCV2022-RIFE ECCV2022 - Real-Time Intermediate Flow Estimation for Video Frame Interpolation 项目地址: https://gitcode.com/gh_mirrors/ec/ECCV2022-RIFE 在当今视频处理领域&am…...

Go-restful容器管理终极指南:多服务部署与负载均衡完整教程

Go-restful容器管理终极指南&#xff1a;多服务部署与负载均衡完整教程 【免费下载链接】go-restful package for building REST-style Web Services using Go 项目地址: https://gitcode.com/gh_mirrors/go/go-restful Go-restful是一个强大的Go语言RESTful Web服务框架…...

技术团队领导一对一沟通指南:打造高效人员管理与反馈机制

技术团队领导一对一沟通指南&#xff1a;打造高效人员管理与反馈机制 【免费下载链接】tlroadmap Тимлид – это ❄️, потому что в каждой компании он уникален и неповторим. 项目地址: https://gitcode.com/gh_m…...