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 文件。
if (mFileType == MediaFile.FILE_TYPE_MP3 ||
mFileType == MediaFile.FILE_TYPE_MP4 ||
mFileType == MediaFile.FILE_TYPE_M4A ||
mFileType == MediaFile.FILE_TYPE_3GPP ||
mFileType == MediaFile.FILE_TYPE_3GPP2 ||
mFileType == MediaFile.FILE_TYPE_OGG ||
mFileType == MediaFile.FILE_TYPE_MID ||
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://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 代码
Java代码
public class
MediaScannerReceiver extends BroadcastReceiver
{
private
final static String TAG =
"MediaScannerReceiver" ;
@Override
public void
onReceive(Context context, Intent intent) {
String action = intent.getAction();
Uri uri = intent.getData();
String externalStoragePath = Environment.getExternalStorageDirectory().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) &&
externalStoragePath.equals(path)) {
scan(context, MediaProvider.EXTERNAL_VOLUME);
} else
if (action.equals(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE) &&
path != null
&& path.startsWith(externalStoragePath +
"/" )) {
scanFile(context, path);
}
}
}
}
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();
args.putString("filepath"
, path);
context.startService(
new
Intent(context, MediaScannerService. class
).putExtras(args));
}
}
6. 根据以上代码得知:
- 当系统启动完毕 会扫描一次
- 当
ACTION_MEDIA_MOUNTED ACTION_MEDIA_SCANNER_SCAN_FILE 也会扫描
7. 如何调用系统MediaScanner 进行扫描
- 通过
Intent.ACTION_MEDIA_MOUNTED 进行全扫描
Java代码
public void
allScan(){
sendBroadcast(new
Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse( "file://"
+ Environment.getExternalStorageDirectory())));
}
- 通过
Intent.ACTION_MEDIA_SCANNER_SCAN_FILE 扫描某个文件
Java代码
public void
fileScan(String fName){
Uri data = Uri.parse("file:///"
+fName);
sendBroadcast(new
Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, data));
}
补充:
上述方法是不支持对文件夹的
即:Uri data 必须是
文件的Uri 如果是文件夹的
其不会起作用的 切记!
- 如何扫描某文件夹下所有文件 难道就不可以么? 当然不 借助于Intent.ACTION_MEDIA_SCANNER_SCAN_FILE
我们可以这么做:
取出该文件夹下的所有子文件
如其是文件且类型符合条件
就取出该文件目录
以 Intent.ACTION_MEDIA_SCANNER_SCAN_FILE方式发送至MediaScannerReceiver
若其为文件夹 则迭代查询之
故实现为:
Java代码
public void
fileScan(String file){
Uri data = Uri.parse("file://"
+file);
Log.d("TAG"
, "file:" +file);
sendBroadcast(new
Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, data));
}
public void
folderScan(String path){
File file = new
File(path);
if
(file.isDirectory()){
File[] array = file.listFiles();
for
( int i= 0 ;i<array.length;i++){
File f = array[i];
if
(f.isFile()){ //FILE TYPE
String name = f.getName();
if
(name.contains( ".mp3" )){
fileScan(f.getAbsolutePath());
}
}
else
{ //FOLDER TYPE
folderScan(f.getAbsolutePath());
}
}
}
}
8. 鉴于多数人并不关心其原理 仅关系如何使用 故 总结如下:
- 扫描全部 我猜测其在效率方面可能有点副作用
Java代码
public void
systemScan(){
sendBroadcast(new
Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse( "file://"
+ Environment.getExternalStorageDirectory())));
}
- 扫描某个文件 参数:填入该文件的路径
Java代码
public void
fileScan(String file){
Uri data = Uri.parse("file://"
+file);
sendBroadcast(new
Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, data));
}
- 扫描文件夹 参数:填入该文件夹路径
Java代码
public void
fileScan(String file){
Uri data = Uri.parse("file://"
+file);
sendBroadcast(new
Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, data));
}
public void
folderScan(String path){
File file = new
File(path);
if
(file.isDirectory()){
File[] array = file.listFiles();
for
( int i= 0 ;i<array.length;i++){
File f = array[i];
if
(f.isFile()){ //FILE TYPE
String name = f.getName();
if
(name.contains( ".mp3" )){
fileScan(f.getAbsolutePath());
}
}
else
{ //FOLDER TYPE
folderScan(f.getAbsolutePath());
}
}
}
}
分享到:
相关推荐
Android多媒体扫描过程[参考].pdf
Android多媒体扫描过程.pdf
Android 多媒体扫描过程详细介绍,MediaScannerReceiver 会在任何的 ACTION_BOOT_COMPLETED, ACTION_MEDIA_MOUNTED 或 ACTION_MEDIA_SCANNER_SCAN_FILE 意图( intent )发出的时候启动。因为解析媒体文件 的元数据 ...
Android多媒体开发高级编程 (Shawn Van Every) pdf扫描版Android多媒体开发高级编程 (Shawn Van Every) pdf扫描版
本书内容翔实、分析深刻,是Android学习多媒体编程不可多得的资料。 来源于网络,版权属于原作者,仅用于学习和交流。
Android多媒体编程从初学到精通 .苗忠良等.扫描版.pdf
《Android多媒体编程从初学到精通].苗忠良等.扫描版.pdf》,值得看看
教程博客在:...Android扫描多媒体文件,Android保存图片到图库,Android保存图片,Android扫描图片到系统图库,Android MediaScannerConnection,Android MediaScannerConnectionClient
主要介绍了Android中扫描多媒体文件操作详解,本文讲解了Android中的多媒体文件扫描机制、如何扫描一个刚创建的文件、如何扫描多个文件,需要的朋友可以参考下
围绕着Android多媒体编程进行,纵向上,从共享内存、IPC通信、HAL、原生服务、JNI调用,以及多媒体的图像、音频、视频框架等多个层次和子系统着手;横向上,贯穿了移动互联网、Android体系应用框架、OpenMAX多媒体...
Android二维码扫描的源码,本源码将演示用手机扫描二维码的整个过程,包括开始扫描,调用扫描框、处理扫描结果等,目前来说,二维码的应用越来越多,用手机扫码现在就是一个基本功能了,因此来说本源码可能会给新手...
三本书 《深入理解Android Wi-Fi、NFC和GPS卷》(邓凡平) [Android开发从入门到精通].扶松柏.扫描版 Android多媒体开发高级编程 以上资料来源于网络如有版权问题请及时联系下架
Android开发精要(完整扫描版)第1章 Android的系统架构/1第2章 Android源代码的下载和编译/16第3章 Android组件模型解析/28第4章 Android的Intent机制/67第5章 组件生命周期解析/91第6章 组件间的数据传输/...
一个对sd卡资源的综合利用,有扫描sd卡,有视频播放和音频播放
第三篇则对三个真实案例的开发步骤进行详细介绍 逐步向读者讲解Android手机应用程序的真实开发过程 同时在源代码中还包含了详细的注释 以尽量帮助读者掌握代码中的每一个细节 尽快掌握Android编程 本书的讲述...
当手机或模拟器开机时,会调用android的MediaScanner,扫描sd卡和内存里的文件。以下是log信息。 代码如下:12-13 15:39:11.062: VERBOSE/MediaPlayerService(67): Create new media retriever from pid 349<BR> 12-...
6.2.2 多媒体框架与实现 /275 6.2.3 android sqlite框架及原理 /285 6.3 扩展库 /289 6.3.1 skia底层库分析 /289 6.3.2 opengl底层库分析 /299 6.3.3 android-openssl实现及运用 /306 6.3.4 freetype及font engine ...