博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Android Media Scanner工作流程及原理
阅读量:4075 次
发布时间:2019-05-25

本文共 10297 字,大约阅读时间需要 34 分钟。

下面是系统 图 

      
1.jpg


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 中列出。下面是支持的文件扩展列表。

[java] 
  1. /* Audio */   
  2.   
  3.   
  4. addFileType("MP3", FILE_TYPE_MP3, "audio/mpeg");   
  5.   
  6.   
  7. addFileType("M4A", FILE_TYPE_M4A, "audio/mp4");   
  8.   
  9.   
  10. addFileType("WAV", FILE_TYPE_WAV, "audio/x-wav");   
  11.   
  12.   
  13. addFileType("AMR", FILE_TYPE_AMR, "audio/amr");   
  14.   
  15.   
  16. addFileType("AWB", FILE_TYPE_AWB, "audio/amr-wb");   
  17.   
  18.   
  19. addFileType("WMA", FILE_TYPE_WMA, "audio/x-ms-wma");   
  20.   
  21.   
  22. addFileType("OGG", FILE_TYPE_OGG, "application/ogg");   
  23.   
  24.   
  25. addFileType("MID", FILE_TYPE_MID, "audio/midi");   
  26.   
  27.   
  28. addFileType("XMF", FILE_TYPE_MID, "audio/midi");   
  29.   
  30.   
  31. addFileType("RTTTL", FILE_TYPE_MID, "audio/midi");   
  32.   
  33.   
  34. addFileType("SMF", FILE_TYPE_SMF, "audio/sp-midi");   
  35.   
  36.   
  37. addFileType("IMY", FILE_TYPE_IMY, "audio/imelody");   
  38.   
  39.   
  40.   
  41.   
  42.   
  43. /* Video */   
  44.   
  45.   
  46. addFileType("MP4", FILE_TYPE_MP4, "video/mp4");   
  47.   
  48.   
  49. addFileType("M4V", FILE_TYPE_M4V, "video/mp4");   
  50.   
  51.   
  52. addFileType("3GP", FILE_TYPE_3GPP, "video/3gpp");   
  53.   
  54.   
  55. addFileType("3GPP", FILE_TYPE_3GPP, "video/3gpp");   
  56.   
  57.   
  58. addFileType("3G2", FILE_TYPE_3GPP2, "video/3gpp2");   
  59.   
  60.   
  61. addFileType("3GPP2", FILE_TYPE_3GPP2, "video/3gpp2");   
  62.   
  63.   
  64. addFileType("WMV", FILE_TYPE_WMV, "video/x-ms-wmv");   
  65.   
  66.   
  67.   
  68.   
  69.   
  70. /* Image */   
  71.   
  72.   
  73. addFileType("JPG", FILE_TYPE_JPEG, "image/jpeg");   
  74.   
  75.   
  76. addFileType("JPEG", FILE_TYPE_JPEG, "image/jpeg");   
  77.   
  78.   
  79. addFileType("GIF", FILE_TYPE_GIF, "image/gif");   
  80.   
  81.   
  82. addFileType("PNG", FILE_TYPE_PNG, "image/png");   
  83.   
  84.   
  85. addFileType("BMP", FILE_TYPE_BMP, "image/x-ms-bmp");   
  86.   
  87.   
  88. addFileType("WBMP", FILE_TYPE_WBMP, "image/vnd.wap.wbmp");   
  89.   
  90.   
  91.   
  92.   
  93.   
  94. /* Audio Play List */   
  95.   
  96.   
  97. addFileType("M3U", FILE_TYPE_M3U, "audio/x-mpegurl");   
  98.   
  99.   
  100. addFileType("PLS", FILE_TYPE_PLS, "audio/x-scpls");   
  101.   
  102.   
  103. addFileType("WPL", FILE_TYPE_WPL, "application/vnd.ms-wpl");   

 

4.Java 层扫描文件


    a ) Java 层开始文件


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


    b)C++ 层扫描文件


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

[javascript] 
  1. if (mFileType == MediaFile.FILE_TYPE_MP3 ||   
  2.   
  3.   
  4. mFileType == MediaFile.FILE_TYPE_MP4 ||   
  5.   
  6.   
  7. mFileType == MediaFile.FILE_TYPE_M4A ||   
  8.   
  9.   
  10. mFileType == MediaFile.FILE_TYPE_3GPP ||   
  11.   
  12.   
  13. mFileType == MediaFile.FILE_TYPE_3GPP2 ||   
  14.   
  15.   
  16. mFileType == MediaFile.FILE_TYPE_OGG ||   
  17.   
  18.   
  19. mFileType == MediaFile.FILE_TYPE_MID ||   
  20.   
  21.   
  22. mFileType == MediaFile.FILE_TYPE_WMA) {   

 

对于被解析的元数据信息, 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://.git.kernel.org/platform/packages/providers/MediaProvider.git

3.  分析源代码:


- Manifest.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 代码


Java代码

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

Java代码


  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 进行全扫描



Java代码

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

Java代码


  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 扫描某个文件


Java代码


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

Java代码


  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   若其为文件夹 则迭代查询之    故实现为:

Java代码

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

Java代码

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


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



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

Java代码


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

Java代码


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


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



Java代码


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

Java代码


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

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

Java代码



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


Java代码


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

转载地址:http://kxuni.baihongyu.com/

你可能感兴趣的文章
StageVideo API
查看>>
[转]三维成像原理
查看>>
Flex Custom Component LifeCycle
查看>>
获取.fla所有导出类名称列表的方法
查看>>
关于FLASH 3D游戏的想法,做一个双人合作射击的游戏,
查看>>
PNG图片优化技术(一)
查看>>
photoshop 优化 PNG 图片尺寸大小 终极秘技!
查看>>
mmo游戏开发应在profile下运行,才能保证正式运行不卡
查看>>
关于Flash CS3创建Sprite类型的问题
查看>>
AS3通俗教程---AS3自身loading制作
查看>>
0 bytes after compression出现的情况
查看>>
内存回收专题
查看>>
[资料] 史上最强的伯克利大学1024线飞龙AI下载地址,有没有人有兴趣来测试一手?...
查看>>
Discuz多人斗地主积分版,消耗论坛积分的斗地主
查看>>
discuz X2斗地主积分版插件安装方法(用户版)
查看>>
ASP.NET程序也能像WinForm程序一样运行
查看>>
听到两个程序员聊天——A:“借我1K块。”
查看>>
轻松搭建一个Windows SVN服务器
查看>>
Discuz X2多人斗地主[消耗论坛积分]小体积版本,仅25MB!
查看>>
大型多人在线MMO RPG游戏最重要的二个职位
查看>>