在做Android媒体应用程序时(Audio、Image、Video)需要对Android的媒体提供者(MediaProvider)做详细的分析,下面记录一下我的收获:

一、获取MediaProvider: 
该工程在系统源码的packages\providers目录下,提出并导入Eclipse,便于阅读;

图中可见都很多报错的,是滴,因为需要一些系统标准sdk之外的接口,不过不影响我们阅读代码。

二、工程结构及内部关系: 
可以从上图看出包含4个文件: 
MediaScannerService.Java:媒体服务,配合广播实现媒体扫描类的实例化,数据库的初始化等工作,也向外提供接口; 
MediaScannerReceiver.Java:一个广播接收器,用于接受系统发给媒体服务的广播并启动媒体服务; 
MediaProvider.Java:媒体数据库的封装类,代码量比较大(四千多行),功能比较复杂,但总的来说就是创建数据库,对外提供URI以实现对数据库的增删改查功能; 
MediaThumbRequest.Java:媒体文件缩略图请求类,与MediaProvider配合使用; 
上一个关系图更直观一些:

上图不是标准的类图,只是为了梳理逻辑关系画的结构图。 
MediaProvider所处的位置及作用见图中红色框中的内容; 
上图还包括其他内容: 
1、App层:audio、image、video如何与媒体库进行交互; 
2、框架层(android.media包下):如何实现媒体的扫描; 
3、Native层:如何实现正在的媒体文件解析; 
4、资源存储层:sd卡、U盘等介质,DTCM存储缩略图;

三、类详解

1、MediaScannerReceiver:

public void onReceive(Context context, Intent intent) {
  String action = intent.getAction();
  Uri uri = intent.getData();
  // String externalStoragePath =
  // Environment.getExternalStorageDirectory().getPath();
  String externalSDStoragePath = Environment
    .getExternalSDStorageDirectory().getPath();
  String externalUDiskStoragePath = Environment
    .getExternalUDiskStorageDirectory().getPath();
  String externalExtSDStoragePath = Environment
    .getExternalExtSDStorageDirectory().getPath();

if (action.equals(Intent.ACTION_BOOT_COMPLETED)) {
   // scan internal storage
   scan(context, MediaProvider.INTERNAL_VOLUME);
  } else {
   if (uri.getScheme().equals("file")) {
    // handle intents related to external storage
    String path = uri.getPath();
    if (action.equals(Intent.ACTION_MEDIA_MOUNTED)) {
     if (externalSDStoragePath.equals(path))
      scan(context, MediaProvider.EXTERNAL_VOLUME_SD);
     else if (externalUDiskStoragePath.equals(path))
      scan(context, MediaProvider.EXTERNAL_VOLUME_UDISK);
     else if (externalExtSDStoragePath.equals(path))
      scan(context, MediaProvider.EXTERNAL_VOLUME_EXTSD);
     else
      Slog.w(TAG, "unknown volume path " + path);
    } else if (action.equals(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE)
      && path != null
      && (path.startsWith(externalSDStoragePath + "/")
        || path.startsWith(externalExtSDStoragePath
          + "/") || path
         .startsWith(externalUDiskStoragePath + "/"))) {
     scanFile(context, path);
    }
   }
  }
 }

三类情况需要启动扫描服务: 
a、系统启动完成; 
b、媒体挂载(EXTERNAL_VOLUME_SD、EXTERNAL_VOLUME_UDISK、EXTERNAL_VOLUME_EXTSD); 
c、媒体文件扫描广播(ACTION_MEDIA_SCANNER_SCAN_FILE);

scanFile和scan方法很简单,只是启动媒体服务即可:

private void scan(Context context, String volume) {
  Bundle args = new Bundle();
  args.putString("volume", volume);
  context.startService(new Intent(context, MediaScannerService.class)
    .putExtras(args));
 }

private void scanFile(Context context, String path) {
  Bundle args = new Bundle();
  Slog.i(TAG, "Start scanFile.");
  args.putString("filepath", path);
  context.startService(new Intent(context, MediaScannerService.class)
    .putExtras(args));
 }

2、MediaScannerService: 
第一步:启动一个线程

public void run() {
  // reduce priority below other background threads to avoid interfering
  // with other services at boot time.
  Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND
    + Process.THREAD_PRIORITY_LESS_FAVORABLE);
  Looper.prepare();

mServiceLooper = Looper.myLooper();
  mServiceHandler = new ServiceHandler();

Looper.loop();
 }

在线程中拿到当前的消息队列,使用handler处理消息;

第二部:启动ServiceHandler处理消息 
ServiceHandler中还是处理两种,一种是扫描,第二种是具体媒体文件的解析; 
看一下第二种是如何实现的:

IBinder binder = arguments.getIBinder("listener");
     IMediaScannerListener listener = (binder == null ? null
       : IMediaScannerListener.Stub.asInterface(binder));
     Uri uri = scanFile(filePath,
       arguments.getString("mimetype"));
     if (listener != null) {
      listener.scanCompleted(filePath, uri);
     }

代码

private final IMediaScannerService.Stub mBinder = new IMediaScannerService.Stub() {
  public void requestScanFile(String path, String mimeType,
    IMediaScannerListener listener) {
   if (Config.LOGD) {
    Log.d(TAG, "IMediaScannerService.scanFile: " + path
      + " mimeType: " + mimeType);
   }
   Bundle args = new Bundle();
   args.putString("filepath", path);
   args.putString("mimetype", mimeType);
   if (listener != null) {
    args.putIBinder("listener", listener.asBinder());
   }
   startService(new Intent(MediaScannerService.this,
     MediaScannerService.class).putExtras(args));
  }

public void scanFile(String path, String mimeType) {
   requestScanFile(path, mimeType, null);
  }
 };

那么问题来了:如果我们在App中想让系统媒体库解析具体某一个文件,应该怎么做呢? 
从上面代码可以看到,MediaScannerService给我们提供的绑定接口,我们只需要传递filepath和一个IMediaScannerListener listener即可,媒体库在解析完之后会回调scanCompleted方法告诉我们解析结果;

第三步:创建MediaScanner对象,完成扫描和解析; 
可见具体扫描、解析工作也不是MediaScannerService做的,MediaScannerService是只在调用sacn、acanfile方法时创建了MediaScanner对象并交给他处理; 
MediaScanner在android.media.MediaScanner系统framework里面,这儿就不做讨论了;

MediaScannerService基本就这些内容了;

3、MediaProvider: 
MediaProvider就是创建数据库,对外提供URI以实现对数据库的增删改查功能;

4、MediaThumbRequest: 
Audio、Image、Video文件都是有缩略图的,缩略图路径存储在DB中,其真实文件存储在sd卡的DICM文件夹下,MediaThumbRequest只是提供给MediaProvider类操作数据库使用。 
主要的就两个方法,一个新建缩略图方法:execute,一个更新缩略图方法:updateDatabase 
新技能get:应用中获取缩略图,期待下一篇文章;

至此,MediaProvider结构分析清楚了,后续计划补两片文章: 
APP中使用系统媒体库; 
媒体文件扫描、解析是如何实现的;

见:

  1. http://www.linuxidc.com/Linux/2015-03/114754.htm
  2. http://www.linuxidc.com/Linux/2015-03/114756.htm

更多Android相关信息见Android 专题页面 http://www.linuxidc.com/topicnews.aspx?tid=11

本文永久更新链接地址http://www.linuxidc.com/Linux/2015-03/114755.htm

Android media媒体库分析之:MediaProvider的更多相关文章

  1. 如何扫描出Android系统媒体库中视频文件

    Android系统启动时会去扫描系统文件,并将系统支持的视频文件(mp4,3gp,wmv)扫描到媒体库(MediaStore)中,下面代码演示如何获得这些文件的信息: publicstatic Lis ...

  2. Android——媒体库 相关知识总结贴

    Android媒体库 http://www.apkbus.com/android-19283-1-1.html Android本地图片选择打开媒体库,选择图片 http://www.apkbus.co ...

  3. Android加入新的视频格式--媒体库扫描

    需求:在mediaprovider数据库中加入.mov后缀格式的视频文件 能够使用工具MediaInfo_GUI_0.7.67_Windows.3243836749.exe 查看mov文件编码格式类型 ...

  4. Android 插入图片到媒体库

    今天介绍一下在Android中怎么插入图片到媒体库,下面看代码: final String titleName = Function.md5(imageUri.toLowerCase()) + &qu ...

  5. Android 获取视频照片与刷新媒体库

    1.获取本地所有视频 public void getLoadMedia() { Cursor cursor = UILApplication.instance.getApplicationContex ...

  6. Android库分析工具(崩溃反编译)

    [时间:2016-07] [状态:Open] [关键词:android, 动态库,静态库, 编译,crash,addr2line] 本文主要整理Android编译系统中可用的库分析工作,可作为后续代码 ...

  7. MTK Android 源码目录分析

    Android 源码目录分析 Android 4.0 |-- abi (application binary interface:应用二进制接口)|-- art (average retrieval ...

  8. Android 的媒体路由功能应用与框架解析

    一.功能描述 Android 的媒体路由API被设计用来允许多种媒体(视频.音乐.图片)在与ANDROID设备连接(无线或有线)的辅助设备(如电视.立体声.家庭戏院系统.音乐播放机)上显示和播放,使用 ...

  9. Android基础-系统架构分析,环境搭建,下载Android Studio,AndroidDevTools,Git使用教程,Github入门,界面设计介绍

    系统架构分析 Android体系结构 安卓结构有四大层,五个部分,Android分四层为: 应用层(Applications),应用框架层(Application Framework),系统运行层(L ...

随机推荐

  1. python paramiko模拟ssh登录,实现sftp上传或者下载文件

    Python Paramiko模块的安装与使用详解 paramiko是短链接,不是持续链接,只能执行你设定的shell命令,可以加分号执行两次命令. http://www.111cn.net/phpe ...

  2. mysql 表连接

    1.子查询是指在另一个查询语句中的SELECT子句. 例句: SELECT * FROM t1 WHERE column1 = (SELECT column1 FROM t2); 其中,SELECT ...

  3. Tomcat服务器顶层结构和启动过程【转】

    号外:2016 最流行的是哪一种 Java 应用服务器呢? 通过从部署的 1240 个 JVM 中得到的数据,我们能够确定出现了 862 个容器供应商,或者说是占到了运行环境的 70% 左右.这些容器 ...

  4. 删除cygwin

    由于cygwin 学习了linux 的用户所有者的方式,要删除有一定困难. 所以要右击  然后点 获取管理员所有权  几分钟之后  就可以删了

  5. C#抓取页面时候,获取页面跳转后的地址

    static string fanhuiurl(string cahxunurl) { string url = ""; HttpWebRequest req = (HttpWeb ...

  6. junit 单元测试 - 参数化测试

    junit4.x版本需要引入如下jar包: hamcrest-core-1.3.jar junit-4.12-beta-3.jar 新建一个计算器类,如下: package com.pt; publi ...

  7. Memcached 缓存服务器介绍

    1.memcached  高性能分布式内存对象缓存系统 2.目的:减轻数据库负载,提高基于动态数据库驱动网站的响应速度 3.数据格式:文本行 4.协议:memcache协议 5.存储方式:hashMa ...

  8. 扩展方法之ToDictionary()

    Person类: public class Person { public int Id { set; get; } public string WorkNo { set; get; } public ...

  9. idea 端口占用

    netstat -anp | grep 8080 lsof -i:8080 查看8080端口被什么进程占用 kill (-s 9) 12903 -s 9 强制 尽快 12903 上面查出的 PID 其 ...

  10. 【搜索 回溯】 zoj 1002

    题意:一些机枪彼此不能在同一行和同一列,但是由于有墙的阻隔,能保证子弹无法穿透,即可以同行同列,现问如果说给了一个n*n(n<=4)的矩阵,并给出了墙的分布情况,能否求出最大能繁殖的机枪数. 思 ...