Android targetSdkVersion 升级到29、30文件处理

4,294 阅读3分钟

最近公布一个非常丰富的开源库,如果你需要相册、录制、录音等操作,那么这个也许对你有一定的帮助: www.jianshu.com/p/8a0accffd…

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",所有目录按照外部存储私有目录来进行读写操作,判断外部存储如果可以操作文件并且存在需要迁移的文件夹,先复制过来,并且需要增加一个全局缓存标记已经进行迁移成功。 如果用到数据库切记先迁移文件后再初始化数据库