1. 应用专属文件
1.1 文件路径
? ? ? ? 应用专属文件的访问与更新不需要任何权限。
????????应用专属文件包括应用内部空间和应用外部空间。以下是内外部空间对应的目录获取方式。其中`getExternalFilesDir`可以设置对应目录下的文件夹,我们可以使用Environment中设置的变量来指定文件夹,防止搞混。
? ? ? ? 这些文件会在应用卸载的时候删除。
? ? ? ? 内部空间中的文件其他应用无法访问。外部空间中的文件,其他应用在适当权限下可以访问。
Context.getExternalCacheDir()
/storage/emulated/0/Android/data/com.example.myapplication/cache
Context.getExternalFilesDir(null)
/storage/emulated/0/Android/data/com.example.myapplication/files
Context.getExternalFilesDir("abcdefg")
/storage/emulated/0/Android/data/com.example.myapplication/files/abcdefg
Context.getCacheDir()
/data/user/0/com.example.myapplication/cache
Context.getFilesDir()
/data/user/0/com.example.myapplication/files
Environment.DIRECTORY_MUSIC = "Music"
Environment.DIRECTORY_PICTURES = "Pictures"
Environment.DIRECTORY_DOWNLOADS = "Download"
Environment.DIRECTORY_DOCUMENTS = "Documents"
Environment.DIRECTORY_SCREENSHOTS = "Screenshots"
Environment.DIRECTORY_MOVIES = "Movies"
? ? ? ? ?还可以获取外部存储的根目录,对文件进行操作。
Environment.getExternalStorageDirectory();
/storage/emulated/0
Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES) // 必须带个目录名字符串
/storage/emulated/0/Pictures
1.2 文件访问
- 使用File API
- context.openFileOutput(filename, Context.MODE_...) 获取FileOutStream
- context.openFileInput() 获取FileInputStream
- 使用文件描述符来访问文件。(如bitmap的生成就可以使用文件描述符)
ParcelFileDescriptor parcelFileDescriptor =
getContentResolver().openFileDescriptor(uri, "r");
FileDescriptor fileDescriptor = parcelFileDescriptor.getFileDescriptor();
2. 共享空间
? ? ? ? 共享空间主要使用ContentResolver + MediaStore API来进行访问。
2.1 权限请求
? ? ? ? 对于Android10(sdk 29) 以上系统并且开启了分区存储的情况下,如果只是读写自身创建的文件,则不需要请求任何权限。如果需要访问其他应用创建的文件,则需要请求以下权限。(一定不要请求WRITE_EXTERNAL_STORAGE权限,因为对于Android10以上的系统并没有这个权限。)
Manifest.permission.READ_EXTERNAL_STORAGE
? ? ? ? 对于Android9及以下的系统,则需要请求下个两个权限。
Manifest.permission.WRITE_EXTERNAL_STORAGE
Manifest.permission.READ_EXTERNAL_STORAGE
?2.2 MediaStoreAPI 访问媒体
2.2.1 查询?
Cursor query = contentResolver.query(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
new String[]{
MediaStore.Images.Media._ID,
MediaStore.Images.Media.DATA,
MediaStore.Images.Media.DISPLAY_NAME,
MediaStore.Images.Media.SIZE
},
MediaStore.Images.Media.DATA + " LIKE ?",
new String[]{"%Screenshot_2022-06-18-10-24%"},
null
);
while (query.moveToNext()) {
String id = query.getString(query.getColumnIndex(MediaStore.Images.Media._ID));
String DATA = query.getString(query.getColumnIndex(MediaStore.Images.Media.DATA));
String DISPLAY_NAME = query.getString(query.getColumnIndex(MediaStore.Images.Media.DISPLAY_NAME));
String EXPOSURE_TIME = query.getString(query.getColumnIndex(MediaStore.Images.Media.SIZE));
uri = ContentUris.withAppendedId(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, Long.parseLong(id));
}
2.2.2? 插入
? ? ? ? 值得注意的是,使用contentResolve.openOutputStream、contentResolve.openInputStream来根据提供的uri来对文件进行读写。
// 1. 插入个空文件占个坑
ContentValues values = new ContentValues();
values.put(MediaStore.Images.Media.DISPLAY_NAME, "hello.jpg");
values.put(MediaStore.Images.Media.AUTHOR, "李嘉浩");
Uri insert = contentResolver.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
try {
// 2. 向文件写入数据
OutputStream outputStream = contentResolver.openOutputStream(insert);
InputStream inputStream = contentResolver.openInputStream(uri); // 这个uri是上面查询部分获取的某一张图片uri。
byte[] buffer = new byte[1024];
int byteNum = 0;
while ((byteNum = inputStream.read(buffer, 0, buffer.length)) != -1) {
outputStream.write(buffer, 0, byteNum);
}
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
参考
官网https://developer.android.com/training/data-storage
|