Android中不同系统的适配一直是应用开发中非常重要的问题,如果不能及时适配不同的系统版本,应用极有可能发生崩溃,给用户带来不好的体验。
随着Google发布 Android Q(API 29),我们需要及时的根据系统的变化做出相应的适配。Android Q中隐私权一块发生了较大的改变。特别是外部存储的访问权限的改变。Android Q中引入了分区存储的概念,应用默认拥有
分区存储的访问(读写)权限,而对于分区存储之外的文件,将无法直接通过文件路径以及File API来进行访问以及操作。
分区存储:
- 应用的私有目录(Context.getExternalFilesDir()目录)
该目录下的文件可以通过文件路径以及File API进行操作,但是应用被卸载时,该目录会被清除 - 应用自己创建的媒体文件(照片,音频,视频)
可以通过MediaStore进行操作
应用访问(读/写)分区存储下的文件不需要再获取任何权限,可以直接访问。
如果应用要访问其它应用创建的文件则必须满足以下两个条件
1.应用已获得READ_EXTERNAL_STORAGE权限
2.文件位于以下其中一个明确的媒体集合中
照片:MediaStore.Image
音频:MediaStore.Audio
视频:MediaStore.Video
为了访问其它应用创建的任何其它文件(包括“downloads”目录下的文件)
“downloads”目录是Android Q才引入的对应MediaStore.Download媒体集合,应用必须使用SAF(存储访问框架),也就是打开系统的文件选择器
可以访问任何文件。
下面我们来具体实践一下Android Q中文件操作;
1.分区存储中应用私有目录中文件的操作
外部存储中应用私有目录就是Context.getExternalFiles(“”)目录
该目录下的文件我们可以使用文件路径和File API进行操作,和以前的文件操作完全一样。
/**
* 创建文件
*/
private void createFile() {
String path = mActivity.getExternalFilesDir("").getAbsolutePath() + File.separator + System.currentTimeMillis() + ".txt";
File file = new File(path);
if (!file.exists()) {
try {
boolean success = file.createNewFile();
Log.e(TAG, "createFile: "+success );
} catch (IOException e) {
e.printStackTrace();
}
}
}
文件的写操作也是一样,通过File拿到文件的输出流,然后进行写操作,这里就不再赘述了
2.应用自己创建的媒体文件或者其他应用创建的媒体文件的操作
如果是自己应用创建的文件,访问无需再获取任何权限,如果是其他应用创建的文件,访问需要获取READ_EXTERNAL_STORAGE权限。操作不再可以使用文件路径和File API进行操作,必须使用MediaStore和ContentResolver来进行文件的访问和操作。
/**
* 新增文件
*/
private void createFile() {
String fileName = System.currentTimeMillis() + "";
ContentValues contentValues = new ContentValues();
contentValues.put(MediaStore.Files.FileColumns.DISPLAY_NAME, fileName);
contentValues.put(MediaStore.Video.VideoColumns.MIME_TYPE, "video/mp4");
Uri uri = mActivity.getContentResolver().insert(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, contentValues);
Log.e(TAG, "createFile: " + uri);
}
上面代码会向数据库里面插入一条记录,会返回一个Uri,但是并不会生成新的文件,只有我们通过
Uri往文件里面写入内容,才会生成文件。
/**
* 查询外部存储所有媒体文件
*/
private void queryFile() {
Cursor cursor = mActivity.getContentResolver().query(MediaStore.Files.getContentUri("external"), null, null, null, null);
while (cursor.moveToNext()) {
String id=cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Files.FileColumns._ID));
String data = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Downloads.DATA));
Log.e(TAG, "queryFile: " + data);
}
}
这里的查询出来的所有的媒体文件的data在Android Q之前是文件的实际路径,而在Android Q中该路径将不再有用,因为我们无法通过文件路径和File API来操作公共目录下的文件。只能通过Uri和ContentResolver来获取文件的输出流来进行操作,而媒体文件的Uri我们可以通过文件的ID来获取
/**
* 根据媒体文件的ID来获取文件的Uri
* @param id
* @return
*/
public String getMediaFileUriFromID(String id) {
return MediaStore.Files.getContentUri("external").buildUpon().appendPath(String.valueOf(id)).build().toString();
}
获取到文件的Uri之后我们就能够通过ContentResolver来获取文件的输出流,来对文件进行写操作
/**
* 对文件进行写操作
* @param uri
*/
private void writeFile(Uri uri) {
try {
AssetFileDescriptor assetFileDescriptor = mActivity.getContentResolver().openAssetFileDescriptor(uri, "rw");
FileOutputStream outputStream = assetFileDescriptor.createOutputStream();
String str = "123";
outputStream.write(str.getBytes());
OutputStreamWriter outputStreamWriter = new OutputStreamWriter(outputStream);
BufferedWriter bufferedWriter = new BufferedWriter(outputStreamWriter);
bufferedWriter.write(str);
bufferedWriter.close();
outputStream.close();
Log.e(TAG, "updateFile: " + uri);
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
/**
* 删除文件
*/
private void deleteFile(Uri uri) {
int deleteNum = mActivity.getContentResolver().delete(uri, null, null);
Log.e(TAG, "deleteFile: " + deleteNum);
}
以上就是分区存储中媒体文件增删改查的操作
3.非私有目录非媒体文件的操作
对于非私有目录,非媒体文件,也就是说如果我们想访问一个外部存储中非私有目录的非媒体文件,或者说是非私有目录的任何类型文件,只能通过SAF(存储访问框架)来进行访问。
可以使用SAF(存储访问框架)访问任何一个文件,而无需请求任何权限。所谓的SAF(存储访问框架)就是
打开系统的文件选择器来进行操作。
/**
* 打开文件选择器选择文件
*/
private void SAF() {
// ACTION_OPEN_DOCUMENT is the intent to choose a file via the system's file
// browser.
Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT);
// Filter to only show results that can be "opened", such as a
// file (as opposed to a list of contacts or timezones)
intent.addCategory(Intent.CATEGORY_OPENABLE);
// Filter to show only images, using the image MIME data type.
// If one wanted to search for ogg vorbis files, the type would be "audio/ogg".
// To search for all documents available via installed storage providers,
// it would be "*/*".
intent.setType("*/*");
startActivityForResult(intent, READ_REQUEST_CODE);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == READ_REQUEST_CODE && resultCode == Activity.RESULT_OK) {
// The document selected by the user won't be returned in the intent.
// Instead, a URI to that document will be contained in the return intent
// provided to this method as a parameter.
// Pull that URI using resultData.getData().
Uri uri = null;
if (data != null) {
uri = data.getData();
Log.e(TAG, "Uri: " + uri.toString());
}
}
}
打开文件选择器选择文件后,会回调onActivityResult,在返回的data中可以拿到文件的Uri,然后对文件进行操作
/**
* 创建文件
* @param mimeType
* @param fileName
*/
private void SAFCreateFile(String mimeType, String fileName) {
Intent intent = new Intent(Intent.ACTION_CREATE_DOCUMENT);
// Filter to only show results that can be "opened", such as
// a file (as opposed to a list of contacts or timezones).
intent.addCategory(Intent.CATEGORY_OPENABLE);
// Create a file with the requested MIME type.
intent.setType(mimeType);
intent.putExtra(Intent.EXTRA_TITLE, fileName);
startActivityForResult(intent, SAF_CREATE_FILE_REQUEST_CODE);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == SAF_CREATE_FILE_REQUEST_CODE && resultCode == Activity.RESULT_OK) {
Uri uri = null;
if (data != null) {
uri = data.getData();
Log.e(TAG, "Uri: " + uri.toString());
}
}
}
打开文件选择器创建好文件后,会回调onActivityResult在返回的data中可以拿到创建的文件的Uri,然后进行
操作。
/**
* 删除文件
*/
private void SAFDeleteFile(Uri uri) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
try {
boolean success = DocumentsContract.deleteDocument(getContentResolver(), uri);
Log.e(TAG, "SAFDeleteFile: " + success);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
}
如果获得了文件的 URI,并且文件的 Document.COLUMN_FLAGS 包含 SUPPORTS_DELETE,则便可通过上述代码删除该文件。
至于通过SAF对文件进行编辑,就是通过SAF选择文件后,拿到文件的Uri,然后和上面对媒体文件进行写操作一样的方式通过文件Uri和ContentResolver拿到文件的输出流,然后进行写操作。
以上就是通过SAF(存储访问框架)来对外部存储中任意类型文件进行增删改查操作。
总结:Android 10的分区存储极大的控制了应用混乱的使用设备的外部存储,从Android 6.0开始的动态权限,Android 7.0的应用私有文件的共享,再到Android 10的分区存储等等关于用户隐私权限的变更,Google无一不在把控制权更多的交给用户,让用户更好的管理和使用自己的设备和应用。