AndroidQ 沙箱適配多媒體文件(小結(jié))
綜述
所有內(nèi)容的訪問(wèn)變化見(jiàn)下圖:

外部媒體文件的掃描,讀取和寫(xiě)入
最容易被踩坑的應(yīng)該是,對(duì)外部媒體文件,照片,視頻,圖片的讀取或?qū)懭搿?/p>
掃描
首先是掃描。掃描依然是使用 query MediaStore 的方式。一句話介紹 MediaStore,MediaStore 就是Android系統(tǒng)中的一個(gè)多媒體數(shù)據(jù)庫(kù)。代碼如下圖所示,以搜索本地視頻為例子:
protected List<VideoInfo> doInBackground(Void... params) {
mContentResolver = context.getContentResolver();
String[] mediaColumns = { MediaStore.Video.Media._ID, MediaStore.Video.Media.DATA,
MediaStore.Video.Media.TITLE, MediaStore.Video.Media.MIME_TYPE,
MediaStore.Video.Media.DISPLAY_NAME, MediaStore.Video.Media.SIZE,
MediaStore.Video.Media.DATE_ADDED, MediaStore.Video.Media.DURATION,
MediaStore.Video.Media.WIDTH, MediaStore.Video.Media.HEIGHT };
Cursor mCursor = mContentResolver.query(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, mediaColumns,
null, null, MediaStore.Video.Media.DATE_ADDED);
if (mCursor == null) {
return null;
}
// 注意,DATA 數(shù)據(jù)在 Android Q 以前代表了文件的路徑,但在 Android Q上該路徑無(wú)法被訪問(wèn),因此沒(méi)有意義。
ixData = mCursor.getColumnIndexOrThrow(MediaStore.Video.Media.DATA);
ixMime = mCursor.getColumnIndexOrThrow(MediaStore.Video.Media.MIME_TYPE);
// ID 是在 Android Q 上讀取文件的關(guān)鍵字段
ixId = mCursor.getColumnIndexOrThrow(MediaStore.Video.Media._ID);
ixSize = mCursor.getColumnIndexOrThrow(MediaStore.Video.Media.SIZE);
ixTitle = mCursor.getColumnIndexOrThrow(MediaStore.Video.Media.TITLE);
allImages = new ArrayList<VideoInfo>();
mTotalVideoCount = 0;
mCursor.moveToLast();
while (mCursor.moveToPrevious()) {
if (addVideo(mCursor) == 0) {
continue;
} else if (addVideo(mCursor) == 1) {
break;
}
}
mCursor.close();
return allImages;
}
既然 data 不可用,就需要知曉 id 的使用方式,首先是使用 id 拼裝出 content uri ,如下所示:
public getRealPath(String id) {
return MediaStore.Video.Media.EXTERNAL_CONTENT_URI.buildUpon().appendPath(String.valueOf(id)).build().toString();
}
Image 同理?yè)Q成 MediaStore.Images。
讀取和寫(xiě)入
其次,是讀取 content uri。這里需要注意 File file = new File(contentUri); 是無(wú)法獲取到文件的。file.exist() 為 false。
那么就產(chǎn)生兩個(gè)問(wèn)題:1. 如何確定 ContentUri 形式的文件存在 2. 如何讀取或?qū)懭胛募?/p>
首先,對(duì)于 Content Uri 的讀取,必須借助于 ContentResolver。
其次,對(duì)于 1,沒(méi)有找到 Google 文檔中提供比較容易的API,只能采用打開(kāi) FileDescriptor 是否成功的形式,代碼如下所示:
public boolean isContentUriExists(Context context, Uri uri) {
if (null == context) {
return false;
}
ContentResolver cr = context.getContentResolver();
try {
AssetFileDescriptor afd = cr.openAssetFileDescriptor(uri, "r");
if (null == afd) {
iterator.remove();
} else {
try {
afd.close();
} catch (IOException e) {
}
}
} catch (FileNotFoundException e) {
return false;
}
return true;
}
這種方法最大的問(wèn)題即是,對(duì)應(yīng)于一個(gè)同步 I/O 調(diào)用,易造成線程等待。因此,目前對(duì)于 MediaStore 中掃描出來(lái)的文件可能不存在的情況,沒(méi)有直接的好方法可以解決過(guò)濾。
對(duì)于問(wèn)題 2,如 1 所示,可以借助 Content Uri 從 ContentResolver 里面拿到 AssetFileDescriptor,然后就可以拿到 InputSteam 或 OutputStream,那么接下來(lái)的讀取和寫(xiě)入就非常自然,如下所示:
public static void copy(File src, ParcelFileDescriptor parcelFileDescriptor) throws IOException {
FileInputStream istream = new FileInputStream(src);
try {
FileOutputStream ostream = new FileOutputStream(parcelFileDescriptor.getFileDescriptor());
try {
IOUtil.copy(istream, ostream);
} finally {
ostream.close();
}
} finally {
istream.close();
}
}
public static void copy(ParcelFileDescriptor parcelFileDescriptor, File dst) throws IOException {
FileInputStream istream = new FileInputStream(parcelFileDescriptor.getFileDescriptor());
try {
FileOutputStream ostream = new FileOutputStream(dst);
try {
IOUtil.copy(istream, ostream);
} finally {
ostream.close();
}
} finally {
istream.close();
}
}
public static void copy(InputStream ist, OutputStream ost) throws IOException {
byte[] buffer = new byte[4096];
int byteCount = 0;
while ((byteCount = ist.read(buffer)) != -1) { // 循環(huán)從輸入流讀取 buffer字節(jié)
ost.write(buffer, 0, byteCount); // 將讀取的輸入流寫(xiě)入到輸出流
}
}
保存媒體文件到公共區(qū)域
這里僅以 Video 示例,Image、Downloads 基本類(lèi)似:
public static Uri insertVideoIntoMediaStore(Context context, String fileName) {
ContentValues contentValues = new ContentValues();
contentValues.put(MediaStore.Video.Media.DISPLAY_NAME, fileName);
contentValues.put(MediaStore.Video.Media.DATE_TAKEN, System.currentTimeMillis());
contentValues.put(MediaStore.Video.Media.MIME_TYPE, "video/mp4");
Uri uri = context.getContentResolver().insert(MediaStore.Video.Media.EXTERNAL_CONTENT_URI, contentValues);
return uri;
}
這里所做的,只是往 MediaStore 里面插入一條新的記錄,MediaStore 會(huì)返回給我們一個(gè)空的 Content Uri,接下來(lái)問(wèn)題就轉(zhuǎn)化為往這個(gè) Content Uri 里面寫(xiě)入,那么應(yīng)用上一節(jié)所述的代碼即可實(shí)現(xiàn)。
Video 的 Thumbnail 問(wèn)題
在 Android Q 上已經(jīng)拿不到 Video 的 Thumbnail 路徑了,又由于沒(méi)有暴露 Video 的 Thumbnail 的 id ,導(dǎo)致了 Video 的 Thumbnail 只能使用實(shí)時(shí)獲取 Bitmap 的方法,如下所示:
private Bitmap getThumbnail(ContentResolver cr, long videoId) throws Throwable {
return MediaStore.Video.Thumbnails.getThumbnail(cr, videoId, MediaStore.Video.Thumbnails.MINI_KIND,
null);
}
可以進(jìn)去看 Android SDK 的實(shí)現(xiàn),其中最關(guān)鍵的部分是:
String column = isVideo ? "video_id=" : "image_id=";
c = cr.query(baseUri, PROJECTION, column + origId, null, null);
if (c != null && c.moveToFirst()) {
bitmap = getMiniThumbFromFile(c, baseUri, cr, options);
if (bitmap != null) {
return bitmap;
}
}
進(jìn)一步再進(jìn)去看,可以發(fā)現(xiàn)直接就把 Video/Image 文件打開(kāi)計(jì)算 Thumbnail。
private static Bitmap getMiniThumbFromFile(
Cursor c, Uri baseUri, ContentResolver cr, BitmapFactory.Options options) {
Bitmap bitmap = null;
Uri thumbUri = null;
try {
long thumbId = c.getLong(0);
String filePath = c.getString(1);
thumbUri = ContentUris.withAppendedId(baseUri, thumbId);
ParcelFileDescriptor pfdInput = cr.openFileDescriptor(thumbUri, "r");
bitmap = BitmapFactory.decodeFileDescriptor(
pfdInput.getFileDescriptor(), null, options);
pfdInput.close();
} catch (FileNotFoundException ex) {
Log.e(TAG, "couldn't open thumbnail " + thumbUri + "; " + ex);
} catch (IOException ex) {
Log.e(TAG, "couldn't open thumbnail " + thumbUri + "; " + ex);
} catch (OutOfMemoryError ex) {
Log.e(TAG, "failed to allocate memory for thumbnail "
+ thumbUri + "; " + ex);
}
return bitmap;
}
這個(gè) API 毫無(wú)疑問(wèn)設(shè)計(jì)的非常不合理,沒(méi)有暴露 Thumbnail 的系統(tǒng)緩存給開(kāi)發(fā)者,造成了每次都要重新I/O 計(jì)算的極大耗時(shí)。強(qiáng)烈呼吁 Android Q 的正式版能修正這個(gè) API 設(shè)計(jì)缺陷。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android?JetPack組件的支持庫(kù)Databinding詳解
DataBinding是Google發(fā)布的一個(gè)數(shù)據(jù)綁定框架,它能夠讓開(kāi)發(fā)者減少重復(fù)性非常高的代碼,如findViewById這樣的操作。其核心優(yōu)勢(shì)是解決了數(shù)據(jù)分解映射到各個(gè)view的問(wèn)題,在MVVM框架中,實(shí)現(xiàn)的View和Viewmode的雙向數(shù)據(jù)綁定2022-08-08
Android實(shí)現(xiàn)點(diǎn)擊Button產(chǎn)生水波紋效果
這篇文章主要介紹了Android實(shí)現(xiàn)點(diǎn)擊Button產(chǎn)生水波紋效果,需要的朋友可以參考下2016-01-01
Android Flutter實(shí)現(xiàn)GIF動(dòng)畫(huà)效果的方法詳解
如果我們想對(duì)某個(gè)組件實(shí)現(xiàn)一組動(dòng)效應(yīng)該怎么辦呢?本文將利用Android Flutter實(shí)現(xiàn)GIF動(dòng)畫(huà)效果,文中的示例代碼講解詳細(xì),需要的可以參考一下2022-06-06
Android編程實(shí)現(xiàn)文件瀏覽功能的方法【類(lèi)似于FileDialog的功能】
這篇文章主要介紹了Android編程實(shí)現(xiàn)文件瀏覽功能的方法,可實(shí)現(xiàn)類(lèi)似于FileDialog的功能,涉及Android針對(duì)文件與目錄操作的相關(guān)技巧,需要的朋友可以參考下2016-11-11
Android StickListView實(shí)現(xiàn)懸停效果
這篇文章主要介紹了Android StickListView實(shí)現(xiàn)懸停效果的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-06-06
android實(shí)現(xiàn)Uri獲取真實(shí)路徑轉(zhuǎn)換成File的方法
這篇文章主要介紹了android實(shí)現(xiàn)Uri獲取真實(shí)路徑轉(zhuǎn)換成File的方法,涉及Android操作路徑的相關(guān)技巧,需要的朋友可以參考下2015-05-05
android簡(jiǎn)單自定義View實(shí)現(xiàn)五子棋
這篇文章主要為大家詳細(xì)介紹了android簡(jiǎn)單自定義View實(shí)現(xiàn)五子棋,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11
Android studio點(diǎn)擊跳轉(zhuǎn)WebView詳解
這篇文章主要為大家詳細(xì)介紹了Android studio點(diǎn)擊跳轉(zhuǎn)WebView的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-09-09
android實(shí)現(xiàn)自動(dòng)發(fā)送郵件
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)自動(dòng)發(fā)送郵件,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07

