转自:http://blog.csdn.net/yan8024/article/details/6620359
下面是系统图
     

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文件。

  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. ……

复制代码

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

c)Java层结束文件

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

5.Java层发送扫描

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

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

MediaScanner

之所以拿MediaScanner开刀 因为想借用系统的Media Scan 工具  通过Intent直接调用系统的

[步骤]

1. 下载并安装Git 过程略 网络上很多

2. 得到该功能的模块地址并使用Git下载之   地址:git://android.git.kernel.org/platform/packages/providers/MediaProvider.git

3.  分析源代码:

- AndroidManifest.xml :  各组件属性描述文件

- MediaProvider : extends ContentProvider  使用SQLiteDatabase 保存查询数据 action="content://media"

- MediaScannerCursor.java

- MediaScannerReceiver : extends BroadcastReceiver   用于接收指定Broadcast: BOOT_COMPLETED MEDIA_MOUNTED MEDIA_SCANNER_SCAN_FILE 并启动 MediaScannerService 开始扫描

- MediaScannerService : extends Service   执行具体的扫描工作

- MediaThumbRequest

4. 鉴于 并不打算自行实现多媒体扫描 因此 此次重点研究对象:MediaScannerReceiver

5. MediaScannerReceiver 代码

  1. public class MediaScannerReceiver extends BroadcastReceiver
  2. {
  3. private final static String TAG = "MediaScannerReceiver";
  4. @Override
  5. public void onReceive(Context context, Intent intent) {
  6. String action = intent.getAction();
  7. Uri uri = intent.getData();
  8. String externalStoragePath = Environment.getExternalStorageDirectory().getPath();
  9. if (action.equals(Intent.ACTION_BOOT_COMPLETED)) {
  10. // scan internal storage
  11. scan(context, MediaProvider.INTERNAL_VOLUME);
  12. } else {
  13. if (uri.getScheme().equals("file")) {
  14. // handle intents related to external storage
  15. String path = uri.getPath();
  16. if (action.equals(Intent.ACTION_MEDIA_MOUNTED) &&
  17. externalStoragePath.equals(path)) {
  18. scan(context, MediaProvider.EXTERNAL_VOLUME);
  19. } else if (action.equals(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE) &&
  20. path != null && path.startsWith(externalStoragePath + "/")) {
  21. scanFile(context, path);
  22. }
  23. }
  24. }
  25. }
  26. private void scan(Context context, String volume) {
  27. Bundle args = new Bundle();
  28. args.putString("volume", volume);
  29. context.startService(
  30. new Intent(context, MediaScannerService.class).putExtras(args));
  31. }
  32. private void scanFile(Context context, String path) {
  33. Bundle args = new Bundle();
  34. args.putString("filepath", path);
  35. context.startService(
  36. new Intent(context, MediaScannerService.class).putExtras(args));
  37. }
  38. }

6.   根据以上代码得知:

-  当系统启动完毕 会扫描一次

-  当 ACTION_MEDIA_MOUNTED ACTION_MEDIA_SCANNER_SCAN_FILE 也会扫描

7.  如何调用系统MediaScanner 进行扫描

- 通过 Intent.ACTION_MEDIA_MOUNTED 进行全扫描

  1. public void allScan(){
  2. sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse("file://"
  3. + Environment.getExternalStorageDirectory())));
  4. }

-  通过 Intent.ACTION_MEDIA_SCANNER_SCAN_FILE 扫描某个文件

  1. public void fileScan(String fName){
  2. Uri data = Uri.parse("file:///"+fName);
  3. sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, data));
  4. }

补充: 上述方法是不支持对文件夹的 即:Uri data 必须是 文件的Uri  如果是文件夹的 其不会起作用的 切记!

- 如何扫描某文件夹下所有文件 难道就不可以么? 当然不 借助于Intent.ACTION_MEDIA_SCANNER_SCAN_FILE

我们可以这么做: 取出该文件夹下的所有子文件 如其是文件且类型符合条件 就取出该文件目录 以 Intent.ACTION_MEDIA_SCANNER_SCAN_FILE方式发送至MediaScannerReceiver   若其为文件夹 则迭代查询之    故实现为:

  1. public void fileScan(String file){
  2. Uri data = Uri.parse("file://"+file);
  3. Log.d("TAG","file:"+file);
  4. sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, data));
  5. }
  6. public void folderScan(String path){
  7. File file = new File(path);
  8. if(file.isDirectory()){
  9. File[] array = file.listFiles();
  10. for(int i=0;i<array.length;i++){
  11. File f = array[i];
  12. if(f.isFile()){//FILE TYPE
  13. String name = f.getName();
  14. if(name.contains(".mp3")){
  15. fileScan(f.getAbsolutePath());
  16. }
  17. }
  18. else {//FOLDER TYPE
  19. folderScan(f.getAbsolutePath());
  20. }
  21. }
  22. }
  23. }

8. 鉴于多数人并不关心其原理 仅关系如何使用 故 总结如下:

-   扫描全部 我猜测其在效率方面可能有点副作用

  1. public void systemScan(){
  2. sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse("file://"
  3. + Environment.getExternalStorageDirectory())));
  4. }

-  扫描某个文件  参数:填入该文件的路径

  1. public void fileScan(String file){
  2. Uri data = Uri.parse("file://"+file);
  3. sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, data));
  4. }

- 扫描文件夹 参数:填入该文件夹路径

  1. public void fileScan(String file){
  2. Uri data = Uri.parse("file://"+file);
  3. sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, data));
  4. }
  5. public void folderScan(String path){
  6. File file = new File(path);
  7. if(file.isDirectory()){
  8. File[] array = file.listFiles();
  9. for(int i=0;i<array.length;i++){
  10. File f = array[i];
  11. if(f.isFile()){//FILE TYPE
  12. String name = f.getName();
  13. if(name.contains(".mp3")){
  14. fileScan(f.getAbsolutePath());
  15. }
  16. }
  17. else {//FOLDER TYPE
  18. folderScan(f.getAbsolutePath());
  19. }
  20. }
  21. }
  22. }

Android之多媒体扫描过程的更多相关文章

  1. Android官方多媒体API Mediacodec翻译(一)

    因近期工作调整,关于Mediacodec部分的翻译会暂停,后续有时间一定补上,非常抱歉. 本文章为根据Android Mediacodec官方英文版的原创翻译,转载请注明出处:http://www.c ...

  2. Android的多媒体框架OpenCore介绍

    网上资料很少, 不过还是找到一个比较详细的说明: 特地在此整理了下: 地址:http://blog.csdn.net/djy1992/article/details/9339787 分为几个阶段: 1 ...

  3. 如何Android中加入扫描名片功能

    要想实现android手机通过扫描名片,得到名片信息,可以使用脉可寻提供的第三方SDK,即Maketion ScanCard SDK,脉可寻云名片识别服务.他们的官方网站为http://www.mak ...

  4. Android开发多媒体应用之SoundPool的使用的代码

    内容过程中,把写内容过程中比较好的内容段记录起来,下面的内容是关于Android开发多媒体应用之SoundPool的使用的内容,希望对各位也有用途. public class MainActivity ...

  5. Android 6.0 扫描不到 Ble 设备需开启位置权限

    Android 6.0 扫描不到 Ble 设备需开启位置权限 之前做 Ble 开发都是在 Android 6.0 系统以下的版本中进行测试的,今天使用 Android 6.0 的设备测试的时候,发现扫 ...

  6. android绘制view的过程

    1 android绘制view的过程简单描述  简单描述可以解释为:计算大小(measure),布局坐标计算(layout),绘制到屏幕(draw):            下面看看每一步的动作到底是 ...

  7. 【转】Android绘制View的过程研究——计算View的大小

    Android绘制View的过程研究——计算View的大小 转自:http://liujianqiao398.blog.163.com/blog/static/18182725720121023218 ...

  8. Android开发——通过扫描二维码,打开或者下载Android应用

    Android开发——通过扫描二维码,打开或者下载Android应用   在实现这个功能的时候,被不同的浏览器折磨的胃疼,最后实现了勉强能用,也查考了一下其他人的博客 android实现通过浏览器点击 ...

  9. Android Activity的启动过程

    文章编辑的太长了,请移步我的csdn博客:http://blog.csdn.net/xyh269 Android Activity的启动过程原文链接:http://blog.csdn.net/xyh2 ...

随机推荐

  1. Metronic 5.0.5 bootstrap后台管理模板

    演示地址:http://keenthemes.com/preview/metronic/    下载 Dashboard Table

  2. Linux 下开发环境的搭建(软件安装)

    一.Centos6.5下安装JDK 步骤1: 查看现有安装的JDK版本 rpm –qa | grep -i java 步骤2: 卸载已有软件 rpm -e --nodeps java-1.7.0-op ...

  3. filebeat安装部署

    简单概述 最近在了解ELK做日志采集相关的内容,这篇文章主要讲解通过filebeat来实现日志的收集.日志采集的工具有很多种,如fluentd, flume, logstash,betas等等.首先要 ...

  4. ThinkPHP join() table()方法的使用,多表查询

    ThinkPHP模型类​比较常用的两个方法,table() join() table 1 $list = M()->table('user1 a, user2 b')->where('a. ...

  5. 300万大奖:欢迎参加美团联合主办的全球AI挑战赛

    2018年8月29日,由美团.创新工场.搜狗.美图联合主办的“AI Challenger 2018全球AI挑战赛”正式启动.美团CTO罗道峰.创新工场CEO李开复.搜狗CEO王小川和美图CEO吴欣鸿共 ...

  6. python获取当前系统的桌面的路径

    一,用内置的winreg(推荐) import winregdef get_desktop():    key = winreg.OpenKey(winreg.HKEY_CURRENT_USER,\  ...

  7. 「WC2016」挑战NPC

    「WC2016」挑战NPC 解题思路 这个题建图非常厉害,带花树什么的只会口胡根本写不动,所以我写了机房某大佬教我的乱搞. 考虑把一个筐 \(x\) 拆成 \(x1,x2,x3\) 三个点,且这三个点 ...

  8. vijos p1876 bfs+map

    题意: Xiaodao是一位喜欢参加ACM比赛的孩子. 所谓ACM比赛, 是一种团队比赛. 每一次比赛, 每队需要由恰好三位选手组成. 现在, Xiaodao希望组建一支新的队伍, 在这之前, 他需要 ...

  9. python开发_xml.etree.ElementTree_XML文件操作_该模块在操作XML数据是存在安全隐患_慎用

    xml.etree.ElementTree模块实现了一个简单而有效的用户解析和创建XML数据的API. 在python3.3版本中,该模块进行了一些修改: xml.etree.cElementTree ...

  10. python开发_logging_日志处理

    在很多编程语言中,都会出现日志处理操作,python也不例外... 接下来我们来看看python中的logging模块 ''' python中,logging模块主要是处理日志的. 所谓日志,可理解为 ...