Chinaunix首页 | 论坛 | 博客
  • 博客访问: 537828
  • 博文数量: 150
  • 博客积分: 5000
  • 博客等级: 大校
  • 技术积分: 1705
  • 用 户 组: 普通用户
  • 注册时间: 2008-11-11 23:29
文章分类

全部博文(150)

文章存档

2011年(9)

2010年(25)

2009年(94)

2008年(22)

我的朋友

分类: LINUX

2010-07-23 17:33:58

下面是系统图 
      

 

 

MediaScannerReceiver会在任何的ACTION_BOOT_COMPLETED, ACTION_MEDIA_MOUNTED或 ACTION_MEDIA_SCANNER_SCAN_FILE 意图(intent)发出的时候启动。因为解析媒体文件的元数据或许会需要很长时间,所以MediaScannerReceiver会启动MediaScannerService。 


MediaScannerService调用一个公用类MediaScanner去处理真正的工作。MediaScannerReceiver维持两种扫描目录:一种是内部卷(internal volume)指向$(ANDROID_ROOT)/media. 另一种是外部卷(external volume)指向$(EXTERNAL_STORAGE). 

扫描和解析工作位于Java层和C++层。JAVA层是启动器。MediaScanner扫描所有目录,如下步骤: 


1.JAVA层初始化 


    在这一步骤中,它会根据目录是在内部卷还是外部卷打开不同的数据库。 


2.Java层预扫描 


    首先清除文件和播放列表的缓存条目。然后根据MediaProvider返回的请求结果生成新文件和播放列表缓存条目。 


3.C++层处理目录 


    列举出所有文件和特定的所有子目录(如果子目录包含一个.nomedia隐藏文件,则不会被列举出来。)。被列举的文件是根据文件扩展来判断文件是否被支持。如果支持这种文件扩展,C++层就会回调到JAVA层扫描文件。这种扩展就会被扫描到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层开始文件 



首先它忽略一些MacOS 和 Windows Media Player特殊的文件。然后它会查看被扫描的文件是否已经存在于缓存条目中,如果存在,它会检查文件上次修改的时间是否改变。最后它返回该文件是否需要进一步处理的结果。如果不需要,接下来的两步不会执行。 



    b)C++层扫描文件 



不是所有的文件都需要交给C++层解析成元数据。只有下面的文件类型会被解析,注意,这里不处理image文件。 

Java代码 
  1. if (mFileType == MediaFile.FILE_TYPE_MP3 ||   
  2. mFileType == MediaFile.FILE_TYPE_MP4 ||   
  3. mFileType == MediaFile.FILE_TYPE_M4A ||   
  4. mFileType == MediaFile.FILE_TYPE_3GPP ||   
  5. mFileType == MediaFile.FILE_TYPE_3GPP2 ||   
  6. mFileType == MediaFile.FILE_TYPE_OGG ||   
  7. mFileType == MediaFile.FILE_TYPE_MID ||   
  8. mFileType == MediaFile.FILE_TYPE_WMA) {   
  9.   
  10.   
  11. ……   
  12.   
  13.   
  14.   
  15. }  


复制代码 

对于被解析的元数据信息,C++层会回调到JAVA层的handleStringTag。Java层会记录它的name/value信息。 


    c)Java层结束文件 


   最后根据上一步解析出的值, Java层会更新相应的MeidaProvider产生的数据库表。 


5.Java层发送扫描 


    到目前为止,所有文件已经被扫描,它最后会检查文件和播放列表缓存条目,看是否所有项仍然存在于文件系统。如果有空条目,则会从数据库中删除。这样它能够保持数据库和文件系统的一致性。 


    其他的应用程序通过接收MediaScannerService发出的ACTION_MEDIA_SCANNER_STARTED 和ACTION_MEDIA_SCANNER_FINISHED意图能够知道什么时候扫描操作开始和结束。

阅读(728) | 评论(0) | 转发(1) |
0

上一篇:[转]VIM快捷键

下一篇:[转]Android应用换肤

给主人留下些什么吧!~~