綜述
所有內容的訪問變化見下圖:
外部媒體文件的掃描,讀取和寫入
最容易被踩坑的應該是,對外部媒體文件,照片,視頻,圖片的讀取或寫入。
掃描
首先是掃描。掃描依然是使用 query MediaStore 的方式。一句話介紹 MediaStore,MediaStore 就是Android系統中的一個多媒體數據庫。代碼如下圖所示,以搜索本地視頻為例子:
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 數據在 Android Q 以前代表了文件的路徑,但在 Android Q上該路徑無法被訪問,因此沒有意義。 ixData = mCursor.getColumnIndexOrThrow(MediaStore.Video.Media.DATA); ixMime = mCursor.getColumnIndexOrThrow(MediaStore.Video.Media.MIME_TYPE); // ID 是在 Android Q 上讀取文件的關鍵字段 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 同理換成 MediaStore.Images。
讀取和寫入
其次,是讀取 content uri。這里需要注意 File file = new File(contentUri); 是無法獲取到文件的。file.exist() 為 false。
那么就產生兩個問題:1. 如何確定 ContentUri 形式的文件存在 2. 如何讀取或寫入文件。
首先,對于 Content Uri 的讀取,必須借助于 ContentResolver。
其次,對于 1,沒有找到 Google 文檔中提供比較容易的API,只能采用打開 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;}
新聞熱點
疑難解答
圖片精選