下面是系統圖
MediaScannerReceiver會(huì )在任何的ACTION_BOOT_COMPLETED, ACTION_MEDIA_MOUNTED或 ACTION_MEDIA_SCANNER_SCAN_FILE 意圖(intent)發(fā)出的時(shí)候啟動(dòng)。因為解析媒體文件的元數據或許會(huì )需要很長(cháng)時(shí)間,所以MediaScannerReceiver會(huì )啟動(dòng)MediaScannerService。
MediaScannerService調用一個(gè)公用類(lèi)MediaScanner去處理真正的工作。MediaScannerReceiver維持兩種掃描目錄:一種是內部卷(internal volume)指向$(ANDROID_ROOT)/media. 另一種是外部卷(external volume)指向$(EXTERNAL_STORAGE).
掃描和解析工作位于
Java層和C++層。JAVA層是啟動(dòng)器。MediaScanner掃描所有目錄,如下步驟:
1.JAVA層初始化
在這一步驟中,它會(huì )根據目錄是在內部卷還是外部卷打開(kāi)不同的數據庫。
2.Java層預掃描
首先清除文件和播放列表的緩存條目。然后根據MediaProvider返回的請求結果生成新文件和播放列表緩存條目。
3.C++層處理目錄
列舉出所有文件和特定的所有子目錄(如果子目錄包含一個(gè).nomedia隱藏文件,則不會(huì )被列舉出來(lái)。)。被列舉的文件是根據文件擴展來(lái)判斷文件是否被支持。如果支持這種文件擴展,C++層就會(huì )回調到JAVA層掃描文件。這種擴展就會(huì )被掃描到MediaFile.java中列出。下面是支持的文件擴展列表。
/* Audio */
addFileType("MP3", FILE_TYPE_MP3, "audio/mpeg");
addFileType("M4A", FILE_TYPE_M4A, "audio/mp4");
addFileType("WAV", FILE_TYPE_WAV, "audio/x-wav");
addFileType("AMR", FILE_TYPE_AMR, "audio/amr");
addFileType("AWB", FILE_TYPE_AWB, "audio/amr-wb");
addFileType("WMA", FILE_TYPE_WMA, "audio/x-ms-wma");
addFileType("OGG", FILE_TYPE_OGG, "application/ogg");
addFileType("MID", FILE_TYPE_MID, "audio/
MIDI");
addFileType("XMF", FILE_TYPE_MID, "audio/midi");
addFileType("RTTTL", FILE_TYPE_MID, "audio/midi");
addFileType("SMF", FILE_TYPE_SMF, "audio/sp-midi");
addFileType("IMY", FILE_TYPE_IMY, "audio/imelody");
/* Video */
addFileType("MP4", FILE_TYPE_MP4, "video/mp4");
addFileType("M4V", FILE_TYPE_M4V, "video/mp4");
addFileType("3GP", FILE_TYPE_3GPP, "video/
3GPP");
addFileType("3GPP", FILE_TYPE_3GPP, "video/3gpp");
addFileType("3G2", FILE_TYPE_3GPP2, "video/3gpp2");
addFileType("3GPP2", FILE_TYPE_3GPP2, "video/3gpp2");
addFileType("WMV", FILE_TYPE_WMV, "video/x-ms-wmv");
/* Image */
addFileType("JPG", FILE_TYPE_JPEG, "image/jpeg");
addFileType("JPEG", FILE_TYPE_JPEG, "image/jpeg");
addFileType("GIF", FILE_TYPE_GIF, "image/gif");
addFileType("PNG", FILE_TYPE_PNG, "image/png");
addFileType("BMP", FILE_TYPE_BMP, "image/x-ms-bmp");
addFileType("WBMP", FILE_TYPE_WBMP, "image/vnd.wap.wbmp");
/* Audio Play List */
addFileType("M3U", FILE_TYPE_M3U, "audio/x-mpegurl");
addFileType("PLS", FILE_TYPE_PLS, "audio/x-scpls");
addFileType("WPL", FILE_TYPE_WPL, "application/vnd.ms-wpl");
4.Java層掃描文件
a)Java層開(kāi)始文件
首先它忽略一些MacOS 和 Windows Media Player特殊的文件。然后它會(huì )查看被掃描的文件是否已經(jīng)存在于緩存條目中,如果存在,它會(huì )檢查文件上次修改的時(shí)間是否改變。最后它返回該文件是否需要進(jìn)一步處理的結果。如果不需要,接下來(lái)的兩步不會(huì )執行。
b)C++層掃描文件
不是所有的文件都需要交給C++層解析成元數據。只有下面的文件類(lèi)型會(huì )被解析,注意,這里不處理image文件。
Java代碼
if (mFileType == MediaFile.FILE_TYPE_MP3 || mFileType == MediaFile.FILE_TYPE_MP4 || mFileType == MediaFile.FILE_TYPE_M4A || mFileType == MediaFile.FILE_TYPE_3GPP || mFileType == MediaFile.FILE_TYPE_3GPP2 || mFileType == MediaFile.FILE_TYPE_OGG || mFileType == MediaFile.FILE_TYPE_MID || mFileType == MediaFile.FILE_TYPE_WMA) { …… }
復制代碼
對于被解析的元數據信息,C++層會(huì )回調到JAVA層的handleStringTag。Java層會(huì )記錄它的name/value信息。
c)Java層結束文件
最后根據上一步解析出的值, Java層會(huì )更新相應的MeidaProvider產(chǎn)生的數據庫表。
5.Java層發(fā)送掃描
到目前為止,所有文件已經(jīng)被掃描,它最后會(huì )檢查文件和播放列表緩存條目,看是否所有項仍然存在于文件系統。如果有空條目,則會(huì )從數據庫中刪除。這樣它能夠保持數據庫和文件系統的一致性。
其他的
應用程序通過(guò)接收MediaScannerService發(fā)出的ACTION_MEDIA_SCANNER_STARTED 和ACTION_MEDIA_SCANNER_FINISHED意圖能夠知道什么時(shí)候掃描操作開(kāi)始和結束。