由于《深入理解Android 卷一》和《深入理解Android卷二》不再出版,而知识的传播不应该由于纸质媒介的问题而中断。所以我将在CSDN博客中全文转发这两本书的全部内容。

第10章 深入理解MediaScanner

本章主要内容

·  介绍多媒体系统中媒体文件扫描的工作原理。

本章涉及的源代码文件名称及位置

以下是本章分析的源代码文件名称及其位置。

·  MediaProvider.java

packages/providers/MediaProvider/MediaProvider.java

·  MediaScannerReceiver.java

packages/providers/MediaProvider/MediaScannerReceiver.java

·  MediaScannerService.java

packages/providers/MediaProvider/MediaScannerService.java

·  MediaScanner.java

framework/base/media/java/com/android/media/MediaScanner.java

·  MediaThumbRequest.java

packages/providers/MediaProvider/MediaThumbRequest.java

·  android_media_MediaScanner.cpp

framework/base/media/jni/android_media_MediaScanner.cpp

·  MediaScanner.cpp

framework/base/media/libmedia/MediaScanner.cpp

·  PVMediasScanner.cpp

external/opencore/android/PVMediasScanner.cpp

10.1  概述

多媒体系统。是Android平台中非常庞大的一个系统。只是由于篇幅所限,本章仅仅介绍多媒体系统中的重要一员MediaScanner。MediaScanner有什么用呢?可能有些读者还不是非常清晰。

MediaScanner和媒体文件扫描有关。比如,在Music应用程序中见到的歌曲专辑名、歌曲时长等信息,都是通过它扫描相应的歌曲而得到的。另外,通过MediaStore接口查询媒体数据库,从而得到系统中全部媒体文件的相关信息也和MediaScanner有关,由于数据库的内容就是由MediaScanner加入的。所以MediaScanner是多媒体系统中非常重要的一部分。

伴随着Android的成长,多媒体系统也发生了非常大的变化。这对开发人员来说,一个非常好的消息。就是从Android 2.3開始那个令人极度郁闷的OpenCore,终于有被干掉的可能了。

从此。也迎来了Stagefright时代。

但Android 2.2在非常长一段时间内还会存在,所以希望以后能有机会深入地剖析这个OpenCore。

以下。就来分析媒体文件扫描的工作原理。

10.2  android.process.media的分析

多媒体系统的媒体扫描功能,是通过一个APK应用程序提供的。它位于package/providers/MediaProvider目录下。通过分析APK的Android.mk文件可知,该APK运行时指定了一个进程名。例如以下所看到的:

application android:process=android.process.media

原来,通过ps命令常常看到的进程就是它啊!另外。从这个APK程序所处的package\providers目录也可知道,它还是一个ContentProvider。事实上从Android应用程序的四大组件来看。它使用了当中的三个组件:

·  MediaScannerService(从Service派生)模块负责扫描媒体文件,然后将扫描得到的信息插入到媒体数据库中。

·  MediaProvider(从ContentProvider派生)模块负责处理针对这些媒体文件的数据库操作请求。比如查询、删除、更新等。

·  MediaScannerReceiver(从BroadcastReceiver派生)模块负责接收外界发来的扫描请求。也就是MS对外提供的接口。

除了支持通过广播发送扫描请求外。MediaScannerService也支持利用Binder机制跨进程调用扫描函数。

这部分内容。将在本章的拓展部分中介绍。

本章仅关注android.process.media进程中的MediaScannerService和MediaScannerReceiver模块。为书写方便起见,将这两个模块简称为MSS和MSR,另外将MediaScanner简称MS。将MediaProvider简称MP。

以下。開始分析android.process.media中和媒体文件扫描相关的工作流程。

10.2.1  MSR模块的分析

MSR模块的核心类MediaScannerReceiver从BroadcastReceiver派生。它是专门用来接收广播的,那么它感兴趣的广播有哪几种呢?其代码例如以下所看到的:

[-->MediaScannerReceiver.java]

public class MediaScannerReceiver extendsBroadcastReceiver

{

private final static String TAG ="MediaScannerReceiver";

@Override  //MSR在onReceive函数中处理广播

publicvoid onReceive(Context context, Intent intent) {

String action = intent.getAction();

Uri uri = intent.getData();

//一般手机外部存储的路径是/mnt/sdcard

String externalStoragePath =

Environment.getExternalStorageDirectory().getPath();

//为了简化书写。全部Intent的ACTION_XXX_YYY字串都会简写为XXX_YYY。

if(action.equals(Intent.ACTION_BOOT_COMPLETED)) {

//假设收到BOOT_COMPLETED广播,则启动内部存储区的扫描工作,内部存储区

//实际上扫描的是/system/media目录。这里存储了系统自带的铃声等媒体文件。

scan(context, MediaProvider.INTERNAL_VOLUME);

}else {

if (uri.getScheme().equals("file")) {

String path = uri.getPath();

/*

注意以下这个推断,假设收到MEDIA_MOUNTED消息。而且外部存储挂载的路径

和“/mnt/sdcard“一样。则启动外部存储也就是SD卡的扫描工作

*/

if (action.equals(Intent.ACTION_MEDIA_MOUNTED) &&

externalStoragePath.equals(path)) {

scan(context,MediaProvider.EXTERNAL_VOLUME);

} else if(action.equals(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE)

&& path != null

&& path.startsWith(externalStoragePath +"/")) {

/*

外部应用能够发送MEDIA_SCANNER_SCAN_FILE广播让MSR启动单个文件

的扫描工作。

注意这个文件必须位于SD卡上。

*/

scanFile(context, path);

}

}

}

}

从上面代码中发现MSR接收的三种请求,也就是说,它对外提供三个接口函数:

·  接收BOOT_COMPLETED请求。这样MSR会启动内部存储区的扫描工作,注意这个内部存储区实际上是/system/media这个目录。

·  接收MEDIA_MOUNTED请求,而且该请求携带的外部存储挂载点路径必须是/mnt/sdcard。通过这样的方式MSR会启动外部存储区也就是SD卡的扫描工作,扫描目标是目录/mnt/sdcard。

·  接收MEDIA_SCANNER_SCAN_FILE请求,而且该请求必须是SD卡上的一个文件,即文件路径须以/mnt/sdcard开头,这样,MSR会启动针对这个文件的扫描工作。

读者是否注意到,MSR和跨Binder调用的接口(在本章拓展内容中将介绍)都不支持对目录的扫描(除了SD卡的根目录外)。实现这个功能并不复杂,有兴趣的读者可自行完毕该功能,假设方便,请将自己实现的代码与大家共享。

大部分的媒体文件都已放在SD卡上了,那么来看收到MEDIA_MOUNTED请求后MSR的工作。

还记得第9章中对Vold的分析吗?这个MEDIA_MOUNTED广播就是由MountService发送的,一旦有SD卡被挂载。MSR就会被这个广播唤醒,接着SD卡的媒体文件就会被扫描了。真是一气呵成!

SD卡根目录扫描时调用的函数scan的代码例如以下:

[-->MediaScannerReceiver.java]

private void scan(Context context, Stringvolume) {

//volume的值为/mnt/sdcard

Bundleargs = new Bundle();

args.putString("volume", volume);

//启动MSS。

context.startService(

new Intent(context, MediaScannerService.class).putExtras(args));

}

scan将启动MSS服务。以下来看MSS的工作。

10.2.2  MSS模块的分析

MSS从Service派生,而且实现了Runnable接口。

以下是它的定义:

[-->MediaScannerService.java]

MediaScannerService extends Service implementsRunnable

//MSS实现了Runnable接口。这表明它可能会创建工作线程

依据SDK中对Service生命周期的描写叙述。Service刚创建时会调用onCreate函数,接着就是onStartCommand函数。之后外界每调用一次startService都会触发onStartCommand函数。接下来去了解一下onCreate函数及onStartCommand函数。

1. onCreate的分析

onCreate函数的代码例如以下所看到的:(这是MSS被系统创建时调用的。在它的整个生命周期内仅调用一次。

[-->MediaScannerService.java]

public void onCreate(){

//获得电源锁,防止在扫描过程中休眠

PowerManager pm = (PowerManager)getSystemService(Context.POWER_SERVICE);

mWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, TAG);

//扫描工作是一个漫长的project。所以这里单独创建一个工作线程,线程函数就是

//MSS实现的Run函数

Threadthr = new Thread(null, this, "MediaScannerService");

thr.start();

|

onCreate将创建一个工作线程:

publicvoid run()

{

/*

设置本线程的优先级,这个函数的调用有非常关键的数据,由于媒体扫描可能会耗费非常长

时间,假设不调低优先级的话。CPU将一直被MSS占用,导致用户感觉系统变得非常慢

*/

Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND +

Process.THREAD_PRIORITY_LESS_FAVORABLE);

Looper.prepare();

mServiceLooper = Looper.myLooper();

/*

创建一个Handler,以后发送给这个Handler的消息都会由工作线程处理。

这一部分内容,已在第5章Handler中分析过了。

*/

mServiceHandler = new ServiceHandler();

Looper.loop();

}

onCreate后,MSS将会创建一个带消息处理机制的工作线程,那么消息是怎么投递到这个线程中的呢?

2. onStartCommand的分析

还记得MSR的scan函数吗?例如以下所看到的:

[-->MediaScannerReceiver.java::scan函数]

context.startService(

new Intent(context, MediaScannerService.class).putExtras(args));

当中Intent包括了目录扫描请求的目标/mnt/sdcard。这个Intent发出后,终于由MSS的onStartCommand收到并处理,其代码例如以下所看到的:

[-->MediaScannerService.java]

@Override

publicint onStartCommand(Intent intent, int flags, int startId)

{

/*

等待mServiceHandler被创建。

耕耘这段代码的码农难道不知道

HandlerThread这个类吗?不熟悉它的读者请再阅读第5章的5.4节。

*/

while(mServiceHandler == null) {

synchronized (this) {

try {

wait(100);

} catch (InterruptedException e) {

}

}

}

......

Message msg = mServiceHandler.obtainMessage();

msg.arg1 = startId;

msg.obj = intent.getExtras();

//往这个Handler投递消息,终于由工作线程处理。

mServiceHandler.sendMessage(msg);

......

}

onStartCommand将把扫描请求信息投递到工作线程去处理。

3. 处理扫描请求

扫描请求由ServiceHandler的handleMessage函数处理。其代码例如以下所看到的:

[-->MediaScannerService.java]

private final class ServiceHandler extendsHandler

{

@Override

public void handleMessage(Message msg)

{

Bundle arguments = (Bundle) msg.obj;

String filePath = arguments.getString("filepath");

try {

......

} else {

String volume =arguments.getString("volume");

String[] directories =null;

if(MediaProvider.INTERNAL_VOLUME.equals(volume)) {

//假设是扫描内部存储的话,实际上扫描的目录是/system/media

directories = newString[] {

Environment.getRootDirectory() + "/media",

};

}

else if (MediaProvider.EXTERNAL_VOLUME.equals(volume)){

//扫描外部存储,设置扫描目标位/mnt/sdcard

directories = new String[]{

Environment.getExternalStorageDirectory().getPath()};

}

if (directories != null) {

/*

调用scan函数开展目录扫描工作。能够一次为这个函数设置多个目标目录,

只是这里仅仅有/mnt/sdcard一个目录

*/

scan(directories, volume);

......

stopSelf(msg.arg1);

}

}

以下,单独用一小节来分析这个scan函数。

4. MSS的scan函数分析

scan的代码例如以下所看到的:

[-->MediaScannerService.java]

private void scan(String[] directories, StringvolumeName) {

mWakeLock.acquire();

ContentValuesvalues = new ContentValues();

values.put(MediaStore.MEDIA_SCANNER_VOLUME, volumeName);

//MSS通过insert特殊Uri让MediaProvider做一些准备工作

UriscanUri = getContentResolver().insert(

MediaStore.getMediaScannerUri(), values);

Uri uri= Uri.parse("file://" + directories[0]);

//向系统发送一个MEDIA_SCANNER_STARTED广播。

sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_STARTED, uri));

try {

//openDatabase函数也是通过insert特殊Uri让MediaProvider打开数据库

if (volumeName.equals(MediaProvider.EXTERNAL_VOLUME)) {

openDatabase(volumeName);

}

//创建媒体扫描器,并调用它的scanDirectories函数扫描目标目录

MediaScanner scanner = createMediaScanner();

scanner.scanDirectories(directories,volumeName);

}

......

//通过特殊Uri让MediaProvider做一些清理工作

getContentResolver().delete(scanUri, null, null);

//向系统发送MEDIA_SCANNER_FINISHED广播

sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_FINISHED, uri));

mWakeLock.release();

}

上面代码中,比較复杂的是MSS和MP的交互。

除了后文中即将看到的正常数据库操作外,MSS还常常会使用一些特殊的Uri来做数据库操作,而MP针对这些Uri会做一些特殊处理,比如打开数据库文件等。

本章不拟对MediaProvider做过多的讨论。这部分知识对那些读完前9章的读者来说,应该不是什么难题。

如有可能,请读者自己整理MediaProvider的工作流程,然后提供给大家一起学习,探讨。

看MSS中创建媒体扫描器的函数createMediaScanner:

private MediaScanner createMediaScanner() {

//以下这个MediaScanner是在framework/base/中,稍后再分析

MediaScanner scanner = new MediaScanner(this);

//获取当前系统使用的区域信息。扫描的时候将把媒体文件里的信息转换成当前系统使用的语言

Locale locale = getResources().getConfiguration().locale;

if(locale != null) {

String language = locale.getLanguage();

String country = locale.getCountry();

String localeString = null;

if (language != null) {

if (country != null) {

//为扫描器设置当前系统使用的国家和语言。

scanner.setLocale(language+ "_" + country);

} else {

scanner.setLocale(language);

}

}

}

return scanner;

}

MSS模块扫描的工作就到此为止了,以下轮到主角MediaScanner登场了。在介绍主角之前,最好还是先总结一下本节的内容。

10.2.3  android.process.media媒体扫描工作的流程总结

媒体扫描工作流程涉及MSR和MSS的交互,来总结一下相关的流程:

·  MSR接收外部发来的扫描请求。并通过startService方式启动MSS处理。

·  MSS的主线程接收MSR所收到的请求。然后投递给工作线程去处理。

·  工作线程做一些前期处理工作后(比如向系统广播扫描開始的消息),就创建媒体扫描器MediaScanner来处理扫描目标。

·  MS扫描完毕后。工作线程再做一些后期处理,然后向系统发送扫描完毕的广播。

10.3  MediaScanner的分析

如今分析媒体扫描器MediaScanner的工作原理,它将纵跨Java层、JNI层。以及Native层。先看它在Java层中的内容。

10.3.1  Java层的分析

1. 创建MediaScanner

认识一下MediaScanner,它的代码例如以下所看到的:

[-->MediaScanner.java]

public class MediaScanner

{

static {

/*

载入libmedia_jni.so。这么重要的库居然放在如此不起眼的MediaScanner类中载入。

个人认为,可能是由于开机后多媒体系统中最先启动的就是媒体扫描工作吧。

*/

System.loadLibrary("media_jni");

native_init();

}

//创建媒体扫描器

public MediaScanner(Context c) {

native_setup();//调用JNI层的函数做一些初始化工作

......

}

在上面的MS中,比較重要的几个调用函数是:

·  native_init和native_setup,关于它们的故事,在分析JNI层时再做介绍。

MS创建好后,MSS将调用它的scanDirectories开展扫描工作,以下来看这个函数。

2. scanDirectories的分析

scanDirectories的代码例如以下所看到的:

[-->MediaScanner.java]

public void scanDirectories(String[]directories, String volumeName) {

try {

long start = System.currentTimeMillis();

initialize(volumeName);//①初始化

prescan(null);//②扫描前的预处理

long prescan = System.currentTimeMillis();

for(int i = 0; i < directories.length; i++) {

/*

③ processDirectory是一个native函数,调用它来对目标目录进行扫描,

当中MediaFile.sFileExtensions是一个字符串,包括了当前多媒体系统所支持的

媒体文件的后缀名。比如.MP3、.MP4等。

mClient为MyMediaScannerClient类型,

它是从MediaScannerClient类派生的。它的作用我们后面再做分析。

*/

processDirectory(directories[i], MediaFile.sFileExtensions,

mClient);

}

long scan = System.currentTimeMillis();

postscan(directories);//④扫描后处理

long end = System.currentTimeMillis();

......//统计扫描时间等

}

上面一共列出了四个关键点,以下逐一对其分析。

(1)initialize的分析

initialize主要是初始化一些Uri。由于扫描时需把文件的信息插入媒体数据库中,而媒体数据库针对Video、Audio、Image文件等都有相应的表,这些表的地址则由Uri表示。

以下是initialize的代码:

[-->MediaScanner.java]

private void initialize(String volumeName) {

//得到IMediaProvider对象,通过这个对象能够对媒体数据库进行操作。

mMediaProvider=

mContext.getContentResolver().acquireProvider("media");

//初始化Uri,以下分别介绍一下。

//音频表的地址。也就是数据库中的audio_meta表。

mAudioUri =Audio.Media.getContentUri(volumeName);

//视频表地址,也就是数据库中的video表。

mVideoUri = Video.Media.getContentUri(volumeName);

//图片表地址,也就是数据库中的images表。

mImagesUri = Images.Media.getContentUri(volumeName);

//缩略图表地址,也就是数据库中的thumbs表。

mThumbsUri = Images.Thumbnails.getContentUri(volumeName);

//假设扫描的是外部存储。则支持播放列表、音乐的流派等内容。

if(!volumeName.equals("internal")) {

mProcessPlaylists = true;

mProcessGenres = true;

mGenreCache = new HashMap<String, Uri>();

mGenresUri = Genres.getContentUri(volumeName);

mPlaylistsUri = Playlists.getContentUri(volumeName);

if ( Process.supportsProcesses()) {

//SD卡存储区域一般使用FAT文件系统,所以文件名称与大写和小写无关

mCaseInsensitivePaths = true;

}

}

}

以下看第二个关键函数prescan。

(2)prescan的分析

在媒体扫描过程中,有个令人头疼的问题。来举个样例,这个样例会贯穿在对这个问题总体分析的过程中。样例:假设某次扫描之前SD卡中有100个媒体文件,数据库中有100条关于这些文件的记录。现因某种原因删除了当中的50个媒体文件,那么媒体数据库什么时候会被更新呢?

读者别小瞧这个问题。如今有非常多文件管理器支持删除文件和目录,它们用起来非常方便,却没有相应地更新数据库。这导致了查询数据库时还能得到这些媒体文件信息,但这个文件实际上已不存在了,而且后面全部和此文件有关的操作都会因此而失败。

事实上,MS已经考虑到这一点了,prescan函数的主要作用是在扫描之前把数据库中和文件相关的信息取出并保存起来,这些信息主要是媒体文件的路径,所属表的Uri。

就上面这个样例来说。它会从数据库中取出100个文件的文件信息。

prescan的代码例如以下所看到的:

[-->MediaScanner.java]

privatevoid prescan(String filePath) throws RemoteException {

Cursor c = null;

String where = null;

String[] selectionArgs = null;

//mFileCache保存从数据库中获取的文件信息。

if(mFileCache == null) {

mFileCache = new HashMap<String, FileCacheEntry>();

}else {

mFileCache.clear();

}

......

try {

//从Audio表中查询当中和音频文件相关的文件信息。

if (filePath != null) {

where = MediaStore.Audio.Media.DATA + "=?";

selectionArgs = new String[] { filePath };

}

//查询数据库的Audio表。获取相应的音频文件信息。

c = mMediaProvider.query(mAudioUri, AUDIO_PROJECTION, where,

selectionArgs,null);

if (c != null) {

try {

while (c.moveToNext()) {

long rowId =c.getLong(ID_AUDIO_COLUMN_INDEX);

//音频文件的路径

String path =c.getString(PATH_AUDIO_COLUMN_INDEX);

long lastModified =

c.getLong(DATE_MODIFIED_AUDIO_COLUMN_INDEX);

if(path.startsWith("/")) {

String key = path;

if(mCaseInsensitivePaths) {

key =path.toLowerCase();

}

//把文件信息存到mFileCache中

mFileCache.put(key,

new FileCacheEntry(mAudioUri, rowId, path,

lastModified));

}

}

} finally {

c.close();

c = null;

}

}

......//查询其它表,取出数据中关于视频,图像等文件的信息并存入到mFileCache中。

finally {

if (c != null) {

c.close();

}

}

}

懂了前面的样例。在阅读prescan函数时可能就比較轻松了。prescan函数运行完后。mFileCache保存了扫描前全部媒体文件的信息。这些信息是从数据库中查询得来的。也就是旧有的信息。

接下来,看最后两个关键函数。

(3)processDirectory和postscan的分析

processDirectory是一个native函数。其具体功能放到JNI层再分析,这里先简介,它在解决上一节那个样例中提出的问题时,所做的工作。

答案是:

processDirectory将扫描SD卡,每扫描一个文件,都会设置mFileCache中相应文件的一个叫mSeenInFileSystem的变量为true。

这个值表示这个文件眼下还存在于SD卡上。

这样,待整个SD卡扫描完后,mFileCache的那100个文件里就会有50个文件的mSeenInFileSystem为true,而剩下的另50个文件则为初始值false。

看到上面的内容,能够知道postscan的作用了吧?就是它把不存在于SD卡的文件信息从数据库中删除,而使数据库得以彻底更新的。来看postscan函数是否是这样处理的:

[-->MediaScanner.java]

private void postscan(String[] directories)throws RemoteException {

Iterator<FileCacheEntry> iterator =mFileCache.values().iterator();

while(iterator.hasNext()) {

FileCacheEntry entry = iterator.next();

String path = entry.mPath;

boolean fileMissing = false;

if (!entry.mSeenInFileSystem) {

if (inScanDirectory(path, directories)) {

fileMissing = true; //这个文件确实丢失了

} else {

File testFile = newFile(path);

if (!testFile.exists()) {

fileMissing = true;

}

}

}

//假设文件确实丢失。则须要把数据库中和它相关的信息删除。

if(fileMissing) {

MediaFile.MediaFileType mediaFileType = MediaFile.getFileType(path);

int fileType = (mediaFileType == null ?

0 : mediaFileType.fileType);

if(MediaFile.isPlayListFileType(fileType)) {

......//处理丢失文件是播放列表的情况

} else {

/*

由于文件信息中还携带了它在数据库中的相关信息,所以从数据库中删除相应的信息会

非常快。

*/

mMediaProvider.delete(ContentUris.withAppendedId(

entry.mTableUri, entry.mRowId), null, null);

iterator.remove();

}

}

}

......//删除缩略图文件等工作

}

Java层中的四个关键点,至此已介绍了三个,另外一个processDirectory是媒体扫描的关键函数,由于它是一个native函数,所以以下将转战到JNI层来进行分析。

10.3.2  JNI层的分析

如今分析MS的JNI层。在Java层中,有三个函数涉及JNI层,它们是:

·  native_init,这个函数由MediaScanner类的static块调用。

·  native_setup。这个函数由MediaScanner的构造函数调用。

·  processDirectory。这个函数由MS扫描目录时调用。

分别来分析它们。

1. native_init函数的分析

以下是native_init相应的JNI函数。其代码例如以下所看到的:

[-->android_media_MediaScanner.cpp]

static void

android_media_MediaScanner_native_init(JNIEnv*env)

{

jclass clazz;

clazz =env->FindClass("android/media/MediaScanner");

//取得Java中MS类的mNativeContext信息。待会创建Native对象的指针会保存

//到JavaMS对象的mNativeContext变量中。

fields.context = env->GetFieldID(clazz,"mNativeContext", "I");

......

}

native_init函数没什么新意,这样的把Native对象的指针保存到Java对象中的做法,已经屡见不鲜。以下看第二个函数native_setup。

2. native_setup函数的分析

native_setup相应的JNI函数例如以下所看到的:

[-->android_media_MediaScanner.cpp]

android_media_MediaScanner_native_setup(JNIEnv*env, jobject thiz)

{

//创建Native层的MediaScanner对象

MediaScanner*mp = createMediaScanner();

......

//把mp的指针保存到Java MS对象的mNativeContext中去

env->SetIntField(thiz,fields.context, (int)mp);

}

//以下的createMediaScanner这个函数将创建一个Native的MS对象

static MediaScanner *createMediaScanner() {

#if BUILD_WITH_FULL_STAGEFRIGHT

charvalue[PROPERTY_VALUE_MAX];

if(property_get("media.stagefright.enable-scan", value, NULL)

&& (!strcmp(value, "1") || !strcasecmp(value,"true"))) {

return new StagefrightMediaScanner; //使用Stagefright的MS

}

#endif

#ifndef NO_OPENCORE

returnnew PVMediaScanner(); //使用Opencore的MS,我们会分析这个

#endif

returnNULL;

}

native_setup函数将创建一个Native层的MS对象。只是可惜的是,它使用的还是Opencore提供的PVMediaScanner,所以后面还不可避免地会和Opencore“正面交锋”。

4. processDirectory函数的分析

看processDirectories函数,它相应的JNI函数代码例如以下所看到的:

[-->android_media_MediaScanner.cpp]

android_media_MediaScanner_processDirectory(JNIEnv*env, jobject thiz,

jstring path, jstring extensions, jobject client)

{

/*

注意上面传入的參数,path为目标目录的路径,extensions为MS支持的媒体文件后缀名集合。

client为Java中的MediaScannerClient对象。

*/

MediaScanner *mp = (MediaScanner*)env->GetIntField(thiz, fields.context);

constchar *pathStr = env->GetStringUTFChars(path, NULL);

constchar *extensionsStr = env->GetStringUTFChars(extensions, NULL);

......

//构造一个Native层的MyMediaScannerClient,并使用Java那个Client对象做參数。

//这个Native层的Client简称为MyMSC。

MyMediaScannerClient myClient(env, client);

//调用Native的MS扫描目录,而且把Native的MyMSC传进去。

mp->processDirectory(pathStr,extensionsStr, myClient,

ExceptionCheck, env);

......

env->ReleaseStringUTFChars(path, pathStr);

env->ReleaseStringUTFChars(extensions,extensionsStr);

......

}

processDirectory函数本身倒不难,但又冒出了几个我们之前没有接触过的类型。以下先来认识一下它们。

5. 究竟有多少种对象?

图10-1展示了MediaScanner所涉及的相关类和它们之间的关系:

图10-1  MS相关类示意图

为了便于理解,便将Java和Native层的对象都画于图中。从上图可知:

·  Java MS对象通过mNativeContext指向Native的MS对象。

·  Native的MyMSC对象通过mClient保存Java层的MyMSC对象。

·  Native的MS对象调用processDirectory函数的时候会使用Native的MyMSC对象。

·  另外。图中Native MS类的processFile是一个虚函数,须要派生类来实现。

当中比較费解的是MyMSC对象。它们有什么用呢?这个问题真是一言难尽。以下通过processDirectory来探寻当中原因。这回得进入PVMediaScanner的领地了。

10.3.3  PVMediaScanner的分析

1. PVMS的processDirectory分析

来看PVMediaScanner(以后简称为PVMS。它就是Native层的MS)的processDirectory函数。这个函数是由它的基类MS实现的。注意。源代码中有两个MediaScanner.cpp,它们的位置各自是:

·  framework/base/media/libmedia

·  external/opencore/android/

看libmedia下的那个MediaScanner.cpp。当中processDirectory函数的代码例如以下所看到的:

[-->MediaScanner.cpp]

status_t MediaScanner::processDirectory(constchar *path,

const char *extensions, MediaScannerClient&client,

ExceptionCheckexceptionCheck, void *exceptionEnv) {

......//做一些准备工作

client.setLocale(locale()); //给Native的MyMSC设置locale信息

//调用doProcessDirectory函数扫描目录

status_tresult =  doProcessDirectory(pathBuffer,pathRemaining,

extensions, client,exceptionCheck, exceptionEnv);

free(pathBuffer);

returnresult;

}

//以下直接看这个doProcessDirectory函数

status_t MediaScanner::doProcessDirectory(char*path, int pathRemaining,

const char *extensions,MediaScannerClient&client,

ExceptionCheck exceptionCheck,void *exceptionEnv) {

......//忽略.nomedia目录

DIR*dir = opendir(path);

......

while((entry = readdir(dir))) {

//枚举目录中的文件和子目录信息

const char* name = entry->d_name;

......

int type = entry->d_type;

......

if(type == DT_REG || type == DT_DIR) {

int nameLength = strlen(name);

bool isDirectory = (type == DT_DIR);

......

strcpy(fileSpot, name);

if (isDirectory) {

......

//假设是子目录。则递归调用doProcessDirectory

int err = doProcessDirectory(path, pathRemaining - nameLength - 1,

extensions, client, exceptionCheck, exceptionEnv);

......

} else if (fileMatchesExtension(path, extensions)) {

//假设该文件是MS支持的类型(依据文件的后缀名来推断)

struct stat statbuf;

stat(path, &statbuf); //取出文件的改动时间和文件的大小

if (statbuf.st_size > 0) {

//假设该文件大小非零,则调用MyMSC的scanFile函数!!?

client.scanFile(path,statbuf.st_mtime, statbuf.st_size);

}

if (exceptionCheck && exceptionCheck(exceptionEnv)) gotofailure;

}

}

}

......

}

假设正在扫描的媒体文件的类型是属于MS支持的,那么,上面代码中最不可思议的是。它居然调用了MSC的scanFile来处理这个文件,也就是说。MediaScanner调用MediaScannerClient的scanFile函数。这是为什么呢?还是来看看这个MSC的scanFile吧。

2. MyMSC的scanFile分析

(1)JNI层的scanFile

事实上。在调用processDirectory时。所传入的MSC对象的真实类型是MyMediaScannerClient,以下来看它的scanFile函数。代码例如以下所看到的:

[-->android_media_MediaScanner.cpp]

virtual bool scanFile(const char* path, longlong lastModified,

long long fileSize)

{

jstring pathStr;

if((pathStr = mEnv->NewStringUTF(path)) == NULL) return false;

//mClient是Java层的那个MyMSC对象。这里调用它的scanFile函数

mEnv->CallVoidMethod(mClient, mScanFileMethodID, pathStr,

lastModified, fileSize);

mEnv->DeleteLocalRef(pathStr);

return (!mEnv->ExceptionCheck());

}

太没有天理了!Native的MyMSCscanFile基本的工作就是调用Java层MyMSC的scanFile函数。这又是为什么呢?

(2)Java层的scanFile

如今仅仅能来看Java层的这个MyMSC对象了,它的scanFile代码例如以下所看到的:

[-->MediaScanner.java]

public void scanFile(String path, longlastModified, long fileSize) {

......

//调用doScanFile函数

doScanFile(path, null, lastModified, fileSize, false);

}

//直接来看doScanFile函数

publicUri doScanFile(String path, String mimeType, long lastModified,

long fileSize, boolean scanAlways) {

/*

上面參数中的scanAlways用于控制是否强制扫描。有时候一些文件在前后两次扫描过程中没有

发生变化,这时候MS能够不处理这些文件。

假设scanAlways为true,则这些没有变化

的文件也要扫描。

*/

Uriresult = null;

long t1 = System.currentTimeMillis();

try{

/*

beginFile的主要工作。就是将保存在mFileCache中的相应文件信息的

mSeenInFileSystem设为true。假设这个文件之前没有在mFileCache中保存,

则会创建一个新项加入到mFileCache中。另外它还会依据传入的lastModified值

做一些处理,以推断这个文件是否在前后两次扫描的这个时间段内被改动。假设有改动,则

须要又一次扫描

*/

FileCacheEntryentry = beginFile(path, mimeType,

lastModified, fileSize);

if(entry != null && (entry.mLastModifiedChanged || scanAlways)) {

String lowpath = path.toLowerCase();

......

if (!MediaFile.isImageFileType(mFileType)) {

//假设不是图片。则调用processFile进行扫描。而图片不须要扫描就能够处理

//注意在调用processFile时把这个Java的MyMSC对象又传了进去。

processFile(path, mimeType, this);

}

//扫描完后,须要把新的信息插入数据库。或者要将原有的信息更新。而endFile就是做这项工作的。

result = endFile(entry, ringtones, notifications,

alarms, music, podcasts);

}

} ......

return result;

}

以下看这个processFile,这又是一个native的函数。

上面代码中的beginFile和endFile函数比較简单,读者能够自行研究。

(3)JNI层的processFile分析

MediaScanner的代码有点绕。是不是?总感觉我们像追兵一样。追着MS在赤水来回地绕。如今应该是二渡赤水了。来看这个processFile函数。代码例如以下所看到的:

[-->android_media_MediaScanner.cpp]

android_media_MediaScanner_processFile(JNIEnv*env, jobject thiz,

jstring path, jstring mimeType, jobject client)

{

//Native的MS还是那个MS,其真实类型是PVMS。

MediaScanner *mp = (MediaScanner *)env->GetIntField(thiz,fields.context);

//又构造了一个新的Native的MyMSC。只是它指向的Java层的MyMSC没有变化。

MyMediaScannerClient myClient(env, client);

//调用PVMS的processFile处理这个文件。

mp->processFile(pathStr,mimeTypeStr, myClient);

}

看来,如今得去看看PVMS的processFile函数了。

3. PVMS的processFile分析

(1)扫描文件

这是我们第一次进入到PVMS的代码中进行分析:

[-->PVMediaScanner.cpp]

status_t PVMediaScanner::processFile(const char*path, const char* mimeType,

MediaScannerClient& client)

{

status_t result;

InitializeForThread();

//调用Native MyMSC对象的函数做一些处理

client.setLocale(locale());

/*

beginFile由基类MSC实现。这个函数将构造两个字符串数组,一个叫mNames,还有一个叫mValues。

这两个变量的作用和字符编码有关,后面会碰到。

*/

client.beginFile();

......

constchar* extension = strrchr(path, '.');

//依据文件后缀名来做不同的扫描处理

if(extension && strcasecmp(extension, ".mp3") == 0) {

result = parseMP3(path, client);//client又传进去了,我们看看对MP3文件的处理

......

}

/*

endFile会依据client设置的区域信息来对mValues中的字符串做语言转换。比如一首MP3

中的媒体信息是韩文,而手机设置的语言为中文简体,endFile会尽量对这些韩文进行转换。

只是语言转换向来是个大难题,不能保证全部语言的文字都能相互转换。转换后的每个value都

会调用handleStringTag做兴许处理。

*/

client.endFile();

......

}

以下再到parseMP3这个函数中去看看。它的代码例如以下所看到的:

[-->PVMediaScanner.cpp]

static PVMFStatus parseMP3(const char *filename,MediaScannerClient& client)

{

//对MP3文件进行解析。得到诸如duration、流派、标题的TAG(标签)信息。

在Windows平台上

//可通过千千静听软件查看MP3文件的全部TAG信息

......

//MP3文件已经扫描完了。以下将这些TAG信息加入到MyMSC中,一起看看

if(!client.addStringTag("duration", buffer))

......

}

(2)加入TAG信息

文件扫描完了。如今须要把文件里的信息通过addStringTag函数告诉给MyMSC。以下来看addStringTag的工作。这个函数由MyMSC的基类MSC处理。

[-->MediaScannerClient.cpp]

bool MediaScannerClient::addStringTag(constchar* name, const char* value)

{

if(mLocaleEncoding != kEncodingNone) {

bool nonAscii = false;

const char* chp = value;

char ch;

while ((ch = *chp++)) {

if (ch & 0x80) {

nonAscii = true;

break;

}

}

/*

推断name和value的编码是不是ASCII,假设不是的话则保存到

mNames和mValues中,等到endFile函数的时候再集中做字符集转换。

*/

if(nonAscii) {

mNames->push_back(name);

mValues->push_back(value);

return true;

}

}

//假设字符编码是ASCII的话。则调用handleStringTag函数。这个函数由子类MyMSC实现。

returnhandleStringTag(name, value);

}

[-->android_media_MediaScanner.cpp::MyMediaScannerClient类]

virtual bool handleStringTag(const char* name,const char* value)

{

......

//调用Java层MyMSC对象的handleStringTag进行处理

mEnv->CallVoidMethod(mClient, mHandleStringTagMethodID, nameStr,valueStr);

}

[-->MediaScanner.java]

publicvoid handleStringTag(String name, String value) {

//保存这些TAG信息到MyMSC相应的成员变量中去。

if (name.equalsIgnoreCase("title") ||name.startsWith("title;")) {

mTitle = value;

} else if (name.equalsIgnoreCase("artist") ||

name.startsWith("artist;")) {

mArtist = value.trim();

} else if (name.equalsIgnoreCase("albumartist") ||

name.startsWith("albumartist;")) {

mAlbumArtist = value.trim();

}

......

}

到这里。一个文件的扫描就算做完了。只是,读者还记得是什么时候把这些信息保存到数据库的吗?

是在Java层MyMSC对象的endFile中。这时它会把文件信息组织起来,然后存入媒体数据库。

10.3.4  MediaScanner的总结

以下总结一下媒体扫描的工作流程,它并不复杂。就是有些绕,如图10-2所看到的:

图10-2  MediaScanner扫描流程图

通过上图能够发现。MS扫描的流程还是比較清晰的。就是四渡赤水这一招,让非常多刚開始学习的人摸不着头脑。

只是读者千万不要像我当初那样。认为这是垃圾代码的代表。

实际上这是码农有意而为之,在MediaScanner.java中通过一段比較具体的凝视,对整个流程做了文字总结,这段总结非常easy。这里就不翻译了。

[-->MediaScanner.java]

//前面还有一段话,读者可自行阅读。

以下是流程的文件总结。

* In summary:

* JavaMediaScannerService calls

* JavaMediaScanner scanDirectories, which calls

* JavaMediaScanner processDirectory (native method), which calls

* nativeMediaScanner processDirectory, which calls

* nativeMyMediaScannerClient scanFile, which calls

* JavaMyMediaScannerClient scanFile, which calls

* JavaMediaScannerClient doScanFile, which calls

* JavaMediaScanner processFile (native method), which calls

* nativeMediaScanner processFile, which calls

* nativeparseMP3, parseMP4, parseMidi, parseOgg or parseWMA, which calls

* nativeMyMediaScanner handleStringTag, which calls

* JavaMyMediaScanner handleStringTag.

* OnceMediaScanner processFile returns, an entry is inserted in to the database.

看完这么具体的凝视。想必你也会认为,码农真是有益这么做的。但他们为什么要设计成这样呢?以后会不会改呢?凝视中也说明了眼下设计的流程是这样,预计以后有可能改。

10.4  拓展思考

10.4.1  MediaScannerConnection介绍

通过前面的介绍,我们知道MSS支持以广播方式发送扫描请求。除了这样的方式外。多媒体系统还提供了一个MediaScannerConnection类,通过这个类能够直接跨进程调用MSS的scanFile。而且MSS扫描完一个文件后会通过回调来通知扫描完毕。MediaScannerConnection类的使用场景包括浏览器下载了一个媒体文件,彩信接收到一个媒体文件等。这时都能够用它来运行媒体文件的扫描工作。

以下来看这个类输出的几个重要API。由于它非常easy。所以这里就不再进行流程的分析了。

[-->MediaScannerConnection.java]

public class MediaScannerConnection implementsServiceConnection {

//定义OnScanCompletedListener接口,当媒体文件扫描完后,MSS调用这个接口进行通知。

publicinterface OnScanCompletedListener {

public void onScanCompleted(String path, Uri uri);

}

//定义MediaScannerConnectionClient接口,派生自OnScanCompletedListener。

//它添加了MediaScannerConnection connect上MSS的通知。

public interface MediaScannerConnectionClient extends

OnScanCompletedListener {

public void onMediaScannerConnected();//连接MSS的回调通知。

public void onScanCompleted(String path, Uri uri);

}

//构造函数。

publicMediaScannerConnection(Context context,

MediaScannerConnectionClient client);

//封装了和MSS连接及断开连接的操作。

publicvoid connect();

publicvoid disconnect()

//扫描单个文件。

publicvoid scanFile(String path, String mimeType);

//我更喜欢以下这个静态函数,它支持多个文件的扫描,实际上间接提供了目录的扫描功能。

publicstatic void scanFile(Context context, String[] paths,

String[] mimeTypes,OnScanCompletedListener callback);

......

}

从使用者的角度来看。本人更喜欢静态的scanFile函数,一方面它封装了和MSS连接等相关的工作,还有一方面它还支持多个文件的扫描,所以如没什么特殊要求,建议读者还是使用这个静态函数。

10.4.2  我问你答

本节是本书的最后一小节,相信一路走来读者对Android的认识和理解也许已有提高。

以下将提几个和媒体扫描相关的问题请读者思考,或者说是提供给读者自行钻研。在解答或研究过程中,读者如有什么心得,最好还是也记录并与我们共享。那些对Android有深刻见地的读者。说不定会收到我们公司HR MM的电话哦!

以下是我在研究MS过程中,认为读者能够进行拓展研究的内容:

·  本书还没有介绍android.process.media中的MediaProvider模块。读者最好还是分别把扫描一个图片、MP3歌曲、视频文件的流程走一遍。只是这个流程分析的重点是MediaProvider。

·  MP中最复杂的是缩略图的生成,读者在完毕上一步的基础上。可集中精力解决缩略图生成的流程。对于视频文件缩略图的生成还会涉及MediaPlayerService。

·  到这一步,相信读者对MP已有了较全面的认识。作为深入学习的跳板,我建议有兴趣的读者能够对Android平台上和数据库有关的模块,以及ContentProvider进行深入研究。这里还会涉及非常多问题,比如query返回的Cursor。是怎么把数据从MediaProvider进程传递到客户端进程的?为什么一个ContentProvider死掉后,它的客户端也会跟着被kill掉?

10.5  本章小结

本章是全书最后一章。也是最轻松的一章。

这一章重点介绍了多媒体系统中和媒体文件扫描相关的知识。相信读者对媒体扫描流程中“四渡赤水”的过程印象会深刻一些。

本章拓展部分介绍了API类MediaScannerConnection的用法。另外,提出了几个和媒体扫描相关的问题请读者与我们共同思考。

[深入理解Android卷一全文-第十章]深入理解MediaScanner的更多相关文章

  1. [深入理解Android卷一全文-第八章]深入理解Surface系统

    由于<深入理解Android 卷一>和<深入理解Android卷二>不再出版.而知识的传播不应该由于纸质媒介的问题而中断,所以我将在CSDN博客中全文转发这两本书的全部内容. ...

  2. [深入理解Android卷一全文-第三章]深入理解init

    因为<深入理解Android 卷一>和<深入理解Android卷二>不再出版,而知识的传播不应该因为纸质媒介的问题而中断,所以我将在CSDN博客中全文转发这两本书的全部内容. ...

  3. [深入理解Android卷一全文-第四章]深入理解zygote

    由于<深入理解Android 卷一>和<深入理解Android卷二>不再出版,而知识的传播不应该由于纸质媒介的问题而中断,所以我将在CSDN博客中全文转发这两本书的所有内容. ...

  4. [深入理解Android卷一全文-第七章]深入理解Audio系统

    由于<深入理解Android 卷一>和<深入理解Android卷二>不再出版,而知识的传播不应该由于纸质媒介的问题而中断,所以我将在CSDN博客中全文转发这两本书的全部内容. ...

  5. 深入理解 Android 之 View 的绘制流程

    概述 本篇文章会从源码(基于Android 6.0)角度分析Android中View的绘制流程,侧重于对整体流程的分析,对一些难以理解的点加以重点阐述,目的是把View绘制的整个流程把握好,而对于特定 ...

  6. 通俗理解Android事件分发与消费机制

    深入:Android Touch事件传递机制全面解析(从WMS到View树) 通俗理解Android事件分发与消费机制 说起Android滑动冲突,是个很常见的场景,比如SliddingMenu与Li ...

  7. [转载] 深入理解Android之Java虚拟机Dalvik

    本文转载自: http://blog.csdn.net/innost/article/details/50377905 一.背景 这个选题很大,但并不是一开始就有这么高大上的追求.最初之时,只是源于对 ...

  8. 理解Android虚拟机体系结构

    1 什么是Dalvik虚拟机 Dalvik是Google公司自己设计用于Android平台的Java虚拟机,它是Android平台的重要组成部分,支持dex格式(Dalvik Executable)的 ...

  9. 深入理解Android之Gradle

    深入理解Android之Gradle 格式更加精美的PDF版请到:http://vdisk.weibo.com/s/z68f8l0xTYrZt 下载 Gradle是当前非常"劲爆" ...

随机推荐

  1. Server Tomcat v8.0 Server at localhost failed to start 问题解决方法?

    bi编程jsp  servlet 第一个程序: HelloServlet 运行错误 404: 十月 28, 2017 11:25:14 上午 org.apache.tomcat.util.digest ...

  2. 关于Adaper的相关用法

    使用BaseAdapter的话需要重载四个方法: getCount getItem getItemId getView getView是用来刷新它所在的ListView的.在每一次item从屏幕外滑进 ...

  3. 一个例子理解ES6的yield关键字

    yield是什么 yield是ES6的新关键字,使函数暂停执行. 一个简单例子 function *countASb() { console.log('Show0:'); var a1 = yield ...

  4. python_文件io

    # -*- coding:UTF-8 -*-#从键盘读入raw_input([prompt]) #函数从标准输入读取一个行,并返回一个字符串(去掉结尾的换行符)#input([prompt]) 函数和 ...

  5. WPF在win7运行时报'Initialization of 'System.Windows.Setter' threw an exception.'

    写的一个WPF程序,在win10运行好好的,在win7就报'Initialization of 'System.Windows.Setter' threw an exception.' 原来是xaml ...

  6. centos7 安装 PostgreSql

    确定你是管理员,然后运行命令: yum -y install postgresql-server postgresql-contrib 初始化数据库 postgresql-setup initdb 启 ...

  7. redis键的过期和内存淘汰策略

    键的过期时间 设置过期时间 Redis可以为存储在数据库中的值设置过期时间,作为一个缓存数据库,这个特性是很有帮助的.我们项目中的token或其他登录信息,尤其是短信验证码都是有时间限制的. 按照传统 ...

  8. 洛谷——P1896 [SCOI2005]互不侵犯

    P1896 [SCOI2005]互不侵犯 状压DP入门题 状压DP一般需要与处理状态是否合法,节省时间 设定状态dp[i][j][k]表示第i行第j个状态选择国王数为k的方案数 $dp[i][j][n ...

  9. 293. [NOI2000] 单词查找树——COGS

    293. [NOI2000] 单词查找树 ★★   输入文件:trie.in   输出文件:trie.out   简单对比时间限制:1 s   内存限制:128 MB 在进行文法分析的时候,通常需要检 ...

  10. chrome本地测试cookie时无效的原因

    https://blog.csdn.net/lyj787505955/article/details/38079079 应该是chrome的原因, 同一网页放在tomcat后,通过localhost方 ...