环境
代码环境:targetSdkVersion 29
手机环境:Android 10
踩坑问题一
场景:api 28升级到api 29,同样的代码从外部存储读写数据,升级后报 open failed: EACCES (Permission denied) 错误。
测试:发现代码运行在Android 10的设备上有问题,运行在Android 10以前的设备没有问题。
解决:在Android使用fuse文件系统开始,Android针对外部存储支持了独立的沙箱存储空间,该空间内的数据为应用独有,并且不需要申请任何权限即可使用。但当时并没有限制应用读写非沙箱内的数据。从Android 10开始,出于数据隐私问题,Android希望禁止应用程序操作非沙箱内的数据。 但是为了过渡,Android提供了requestLegacyExternalStorage机制,来帮助应用使用原来的机制继续读写存储卡。 关于Android 存储机制的说明请参考Android官方文档。
为了过渡,Android提供了以下策略:
- targetSdkVersion 29的Android应用程序在获得读写存储卡权限的情况下,必须在AndroidManifest.xml的application标签下声明requestLegacyExternalStorage=true,才可以访问沙箱路径下的数据。
- targetSdkVersion < 29的Android应用程序默认带有requestLegacyExternalStorage=true属性。