code小生 一个专注大前端领域的技术平台公众号回复Android
加入安卓技术群
作者:zhongjh
链接:https://www.jianshu.com/p/892a2ca5c41e
声明:本文已获zhongjh
授权发表,转发等请联系原作者授权
android12要来了,很多老app还在29以下,该文章就是讲述升级到29、30关于文件的处理。
在总结多种情况之前,我们再次确认下目前有哪些存储目录
私有存储 (Private Storage) : 每个应用在都拥有自己的私有目录,其它应用看不到,彼此也无法访问到该目录。
内部存储私有目录 (/data/data/packageName) ;
外部存储私有目录 (/sdcard/Android/data/packageName);
共享存储 (Shared Storage) : 存储其他应用可访问文件, 包含媒体文件、文档文件以及其他文件,对应设备DCIM、Pictures、Alarms、Music、Notifications、Podcasts、Ringtones、Movies、Download等目录。
外部存储:Environment.getExternalStorageDirectory()获取sdcard下的任意文件夹,在SDK29以上已经过期、失效。
Android版本迭代变化
在29版本后,只能操作本身内部存储私有目录、外部存储私有目录、共享存储,但是依然可以通过android:requestLegacyExternalStorage="true"来设置(在AndroidManifest.xml中的application添加该配置),不启用分区存储,一切照旧。
但是30版本以后,就强制性的只能操作规定的目录,这个时候依然有个兼容配置设置,android:preserveLegacyExternalStorage="true"
(在AndroidManifest.xml中的application添加该配置),这个配置使得手机appSdk30版本以下,更新appSdk30版本以后,依然不启用分区存储,一切照旧。
那么我简单总结多种情况和解决方式:
targetSdkVersion = 28,运行后正常读写所有文件,如果不是必须的需求并且是新创建的项目的话,建议把文件按照规范存储在外部存储私有目录 (/sdcard/Android/data/packageName)
targetSdkVersion = 29,targetSdkVersion 由 低版本 修改到 29,覆盖安装,运行后正常读写。
targetSdkVersion = 29,卸载旧应用,重新安装新应用,如果读写外部存储,程序崩溃 (open failed: EACCES (Permission denied))
targetSdkVersion = 29,添加android:requestLegacyExternalStorage="true"(不启用分区存储),读写正常不报错
targetSdkVersion = 30,targetSdkVersion 由 低版本 修改到 30,覆盖安装,读写报错,程序崩溃 (open failed: EACCES (Permission denied))
targetSdkVersion = 30,targetSdkVersion 由 低版本 修改到 30,覆盖安装,增加 android:preserveLegacyExternalStorage="true",读写正常不报错
targetSdkVersion = 30,卸载旧应用,重新安装新应用,不管设置任何配置,如果读写外部存储,程序崩溃 (open failed: EACCES (Permission denied))
如果我app缓存文件存储在外部存储,那么如何处理升级到30并且迁移文件呢?
客户app有重要文件在外部存储
客户app要升级sdk到30
要考虑新客户安装app
所以最终是设置android:preserveLegacyExternalStorage="true"
,所有目录按照外部存储私有目录来进行读写操作,判断外部存储如果可以操作文件并且存在需要迁移的文件夹,先复制过来,并且需要增加一个全局缓存标记已经进行迁移成功。
如果用到数据库切记先迁移文件后再初始化数据库。
相关推荐
细细品读!深入浅出,官方文档看ConstraintLayout
我是code小生
,喜欢可以随手点个在看
、转发给你的朋友,谢谢~
如果你有写博客的好习惯
欢迎投稿
点个“在看”小生感恩❤️