下面是系統(tǒng)圖
MediaScannerReceiver會在任何的ACTION_BOOT_COMPLETED, ACTION_MEDIA_MOUNTED或 ACTION_MEDIA_SCANNER_SCAN_FILE 意圖(intent)發(fā)出的時(shí)候啟動。因?yàn)榻馕雒襟w文件的元數(shù)據(jù)或許會需要很長時(shí)間,所以MediaScannerReceiver會啟動MediaScannerService。
MediaScannerService調(diào)用一個(gè)公用類MediaScanner去處理真正的工作。MediaScannerReceiver維持兩種掃描目錄:一種是內(nèi)部卷(internal volume)指向$(ANDROID_ROOT)/media. 另一種是外部卷(external volume)指向$(EXTERNAL_STORAGE).
掃描和解析工作位于
Java層和C++層。JAVA層是啟動器。MediaScanner掃描所有目錄,如下步驟:
1.JAVA層初始化
在這一步驟中,它會根據(jù)目錄是在內(nèi)部卷還是外部卷打開不同的數(shù)據(jù)庫。
2.Java層預(yù)掃描
首先清除文件和播放列表的緩存條目。然后根據(jù)MediaProvider返回的請求結(jié)果生成新文件和播放列表緩存條目。
3.C++層處理目錄
列舉出所有文件和特定的所有子目錄(如果子目錄包含一個(gè).nomedia隱藏文件,則不會被列舉出來。)。被列舉的文件是根據(jù)文件擴(kuò)展來判斷文件是否被支持。如果支持這種文件擴(kuò)展,C++層就會回調(diào)到JAVA層掃描文件。這種擴(kuò)展就會被掃描到MediaFile.java中列出。下面是支持的文件擴(kuò)展列表。
/* 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層開始文件
首先它忽略一些MacOS 和 Windows Media Player特殊的文件。然后它會查看被掃描的文件是否已經(jīng)存在于緩存條目中,如果存在,它會檢查文件上次修改的時(shí)間是否改變。最后它返回該文件是否需要進(jìn)一步處理的結(jié)果。如果不需要,接下來的兩步不會執(zhí)行。
b)C++層掃描文件
不是所有的文件都需要交給C++層解析成元數(shù)據(jù)。只有下面的文件類型會被解析,注意,這里不處理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) { …… }
復(fù)制代碼
對于被解析的元數(shù)據(jù)信息,C++層會回調(diào)到JAVA層的handleStringTag。Java層會記錄它的name/value信息。
c)Java層結(jié)束文件
最后根據(jù)上一步解析出的值, Java層會更新相應(yīng)的MeidaProvider產(chǎn)生的數(shù)據(jù)庫表。
5.Java層發(fā)送掃描
到目前為止,所有文件已經(jīng)被掃描,它最后會檢查文件和播放列表緩存條目,看是否所有項(xiàng)仍然存在于文件系統(tǒng)。如果有空條目,則會從數(shù)據(jù)庫中刪除。這樣它能夠保持?jǐn)?shù)據(jù)庫和文件系統(tǒng)的一致性。
其他的
應(yīng)用程序通過接收MediaScannerService發(fā)出的ACTION_MEDIA_SCANNER_STARTED 和ACTION_MEDIA_SCANNER_FINISHED意圖能夠知道什么時(shí)候掃描操作開始和結(jié)束。