前言
由于Google对用户隐私和系统安全做的越来越完善,应用对一些敏感信息的操作越来越难。比如最常见的共享存储空间的访问,像保存图片到相册这种常见的需求。
Android 6.0
以前,应用要想保存图片到相册,只需要通过File
对象打开IO流就可以保存;Android 6.0
添加了运行时权限,需要先申请存储权限才可以保存图片;Android 10
引入了分区存储,但不是强制的,可以通过清单配置android:requestLegacyExternalStorage="true"
关闭分区存储;Android 11
强制开启分区存储,应用以 Android 11 为目标版本,系统会忽略requestLegacyExternalStorage
标记,访问共享存储空间都需要使用MediaStore
进行访问。
注意
- 使用内部存储应用卸载后,存储的文件自动删除
- 使用外部存储要申请权限,并且要判断外部存储是否可用
申请权限
有些权限必须用户同意才能调用相应功能,所以开发者需要调用权限申请的代码,弹出一个小窗口,向用户动态申请权限。
首先要在AndroidManifest.xml文件的application 标签下 加一条属性 android:requestLegacyExternalStorage="true"
还有读写权限:
1 | <!--Android Q之后不需要存储权限,完全使用MediaStore API来实现--> |
以下是动态申请文件读写权限的过程:
1.读与写的权限先定义到静态字符数组中:
1 | private static String[] PERMISSIONS_STORAGE = { |
2.首先判断当前系统是否是Android6.0(对应API 23)以及以上,如果是则判断是否含有了写文件的权限,如果没有则调用动态申请权限的代码,ActivityCompat.requestPermission方法的第一个参数是目标Activity,填写this即可,第二个参数是String[]字符数组类型的权限集,第三个即请求码:
1 | if (Build.VERSION.SDK_INT > Build.VERSION_CODES.LOLLIPOP) { |
3.回调函数,申请权限后回调onRequestPermissionResult函数,第一个参数为请求码,第二个参数是刚刚请求的权限集,第三个参数是请求结果,0表示授权成功,-1表示授权失败:
1 |
|
4.所有代码:
1 | import androidx.annotation.NonNull; |
文件存储
要求文件公开
1 | public static String getFilePath(Context context, String dir) { |
要求文件私有的话
1 | public static String getFilePath(Context context, String dir) { |
内部存储
context.getCacheDir
1 | File cacheDir = context.getCacheDir(); |
特点
- 应用内部存储空间
- 数据文件私有
- 不需要申请权限
- 当应用被卸载的时候,目录下的文件会被删除
- 存储小文件
需要注意的是,这个文件的目录和应用的存储位置有关,
当应用被移动到外部存储设备的时候,文件的绝对路径也是变化的,所以建议当数据存储到这个目录的时候,用相对路径。
这个目录和getFilesDir()
目录最大的不同在于:当安卓设备的存储空间少,或者不够用的时候,系统会自动删除这个目录下的文件。
官方建议是,超过1MB的文件,建议存储到
getExternalCacheDir()
目录下
context.getFilesDir
1 | File filesDir = context.getFilesDir(); |
特点
- 应用内部存储空间
- 数据文件私有
- 不需要申请权限
- 当应用被卸载的时候,目录下的文件会被删除
当应用被移动到外部存储设备的时候,文件的绝对路径也是变化的,所以建议当数据存储到这个目录的时候,用相对路径
系统提供的访问此路径文件的方法是:
1 | context.openFileOutput(String,int); |
外部存储
Environment.getExternalStoragePublicDirectory
1 | File externalStoragePublicDirectory = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES); |
特点
- 应用外部存储空间
- 数据文件非私有
- 需要申请权限(
READ_EXTERNAL_STORAGE
,WRITE_EXTERNAL_STORAGE
) - 需要判断外部存储的挂载状态(
getExternalStorageState(File)
) - 当应用被卸载的时候,目录下的不会被删除
这个目录是用来存放各种类型的文件的目录,在这里用户可以分类管理不同类型的文件(例如音乐、图片、电影等)
类型如下:
- DIRECTORY_MUSIC
- DIRECTORY_PODCASTS
- DIRECTORY_RINGTONES
- DIRECTORY_ALARMS
- DIRECTORY_NOTIFICATIONS
- DIRECTORY_PICTURES
- DIRECTORY_MOVIES
- DIRECTORY_DOWNLOADS
- DIRECTORY_DCIM
- DIRECTORY_DOCUMENTS
Environment.getExternalStorageDirectory
1 | File externalStorageDirectory = Environment.getExternalStorageDirectory(); |
特点
- 应用外部存储空间
- 数据文件非私有
- 需要申请权限(
READ_EXTERNAL_STORAGE
,WRITE_EXTERNAL_STORAGE
) - 需要判断外部存储的挂载状态(
getExternalStorageState(File)
) - 当应用被卸载的时候,目录下的不会被删除
注:在该目录下读写文件,需要获取读写权限
该目录下的文件,这个目录是用户进行操作的一个根目录,进入二级目录可以通过getExternalFilesDirs(String)
, getExternalCacheDirs()
, and getExternalMediaDirs()
这些方法
官方建议:
不要直接使用该目录,为了避免污染用户的根命名空间
应用私有的数据,应该放在context.getExternalFilesDir
目录下
其他的可以被分享的文件,可以放在Environment.getExternalStoragePublicDirectory(String)
目录下
context.getExternalFilesDir
1 | File externalFilesDir = context.getExternalFilesDir(null); |
特点
- 应用外部存储空间
- 数据文件私有
- 需要申请权限(
READ_EXTERNAL_STORAGE
,WRITE_EXTERNAL_STORAGE
) - 需要判断外部存储的挂载状态(
getExternalStorageState(File)
) - 当应用被卸载的时候,目录下的文件会被删除
当应用被卸载的时候,目录下的文件会被删除,但是这里和getFilesDir()
还有不同之处:
只有手机系统使用的是虚拟外部存储(虚拟SD卡)的时候,才可以在卸载应用的同时,自动删除该目录下的文件,如果是之前的物理存储(物理SD卡)则不会自动删除该目录,及目录下的文件
context.getExternalCacheDir
1 | File externalCacheDir = context.getExternalCacheDir(); |
特点
- 应用外部存储空间
- 数据文件私有
- 需要申请权限(
READ_EXTERNAL_STORAGE
,WRITE_EXTERNAL_STORAGE
) - 需要判断外部存储的挂载状态(
getExternalStorageState(File)
) - 当应用被卸载的时候,目录下的文件会被删除
- 存储大文件
当应用被卸载的时候,目录下的文件会被删除,但是这里和getCacheDir()
还有不同之处:
只有手机系统使用的是虚拟外部存储(虚拟SD卡)的时候,才可以在卸载应用的同时,自动删除该目录下的文件
如果是之前的物理存储(物理SD卡)则不会自动删除该目录,及目录下的文件