1.下面是系统 图
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));
}
}
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 进行全扫描
public void allScan(){
sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse( "file://"
+ Environment.getExternalStorageDirectory())));
}
- 通过 Intent.ACTION_MEDIA_SCANNER_SCAN_FILE 扫描某个文件
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 若其为文件夹 则迭代查询之 故实现为:
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());
}
}
}
}
- 大小: 31.4 KB
分享到:
相关推荐
在Android系统中,多媒体扫描过程是一个关键的组件,它负责识别和处理设备上的音频、视频和图像文件。这个过程涉及到多个层次的交互,包括BroadcastReceiver、Service和JNI层的配合。下面将详细解释这一过程: 1. *...
Android多媒体扫描过程是一个自动化的服务,用于检测、解析并更新设备上的多媒体文件信息。这个过程由多个组件协同完成,确保系统能够正确地识别和组织这些文件。以下是详细的解释: 1. **触发扫描的事件**: - ...
ype("SMF", FILE_TYPE_MID, "audio/spm"); addFileType("IMY", FILE_TYPE_MID, "audio/imy"); addFileType("MXMF", FILE_TYPE_MID, "audio/midi"); addFileType("RM", FILE_TYPE_RMI, "audio/x-pn-realaudio-plugin...
Android多媒体扫描框架,即Media Scanner,是Android系统中用于检测和处理媒体文件(如音频、视频和图片)的重要组件。当设备启动、媒体设备挂载或接收到特定广播意图时,MediaScannerReceiver会被触发,进而启动...
'android.intent.action.MEDIA_SCANNER_FINISHED' 开始扫描介质的一个目录。 'android.intent.action.MEDIA_SCANNER_STARTED' 扩展介质的挂载被解除 (unmount)。 因为它已经作为 USB 大容量存储被共享。 'android....
2. **媒体扫描(Media Scanner)**:Android系统提供媒体扫描服务,可以扫描USB设备上的媒体文件(音频和视频)。你可以通过`MediaScannerConnection`类来调用此服务,将新插入的USB设备中的媒体文件添加到系统的...
通过这个类,开发者可以控制扫描过程,如设置解码类型(条形码、二维码等)、扫描区域、照明模式等。 2. **ScannerView**:这是一个自定义视图,用于显示扫描的预览画面。它可以处理相机权限、相机预览流的处理,...
在Android开发中,集成二维码和条形码扫描功能是一项常见的需求。Zxing(又名ZXing,意为“zebra crossing”)是一个开源项目,提供了跨平台的多种条码读取功能,包括二维码和条形码。在这个项目中,我们将讨论如何...
【标题】"Delphi XE7 Android二维码扫描ZXing"涉及的是在Delphi XE7环境下,使用ZXing库开发Android应用,实现二维码和条形码的扫描功能。ZXing,全称为“Zebra Crossing”,是一款开源的多平台条码读取库,支持多种...
它定义了一套规则和标准,使得软件能够模拟真实扫描仪的行为,包括设置扫描参数(如分辨率、色彩模式)、控制扫描过程、以及处理扫描后的图像数据。虚拟扫描仪协议还可能包括对图像质量的优化、格式转换、网络传输等...
驱动程序-扫描仪驱动-明基benq scanner 5000s扫描仪驱动.zip
闪光灯的使用则是在二维码扫描过程中提高扫描效果的一个关键因素,特别是在光线不足的环境下。在Android中,可以通过Camera API或Camera2 API来控制闪光灯。以下是使用Camera API开启和关闭闪光灯的示例代码: ```...
对于横竖屏切换,我们可以通过在`AndroidManifest.xml`中为扫描Activity添加`android:configChanges="orientation|screenSize"`,并覆盖`onConfigurationChanged`方法,避免Activity被重新创建,保证扫描过程的连续...
在扫描过程中,`MediaScanner`会解析多媒体文件的元数据,如文件名、大小、创建日期、修改日期、专辑、艺术家等,并将这些信息存储到MediaStore数据库中。这样,其他应用可以通过查询MediaStore来获取和展示这些...
根据ZXing 3.1.0 源码编译,未做任何发动。
1.更新多媒体数据库,类似Media Scanner 2.通过MediaStore 获取多媒体的meta 数据。
在Android开发中,二维码的扫描和生成是常见的需求,用于数据快速交换和识别。ZXing(Zebra Crossing)是一个开源的、跨平台的条码处理库,它支持多种类型的条码,包括一维码和二维码,如QR码。在本项目中,我们将...
在Android的原生多媒体框架中,mediaScanner的JNI实现通常位于`frameworks/base/media/jni/`目录下,文件名为`android_media_MediaScanner.cpp`。这里定义了native方法的实现,负责将Java层的扫描任务委托给底层的...
基于barcodescanner实现Android二维码扫描功能 Android二维码扫描功能是当前移动应用程序中非常常见的一种基础功能。随着二维码技术的普及,二维码扫描功能已经成为许多应用程序的必备功能之一。在Android平台上...
"安卓串口Socket通讯USB驱动jni相关-AndroidScanner扫描目标设备可用端口的小程序.rar"是一个针对这一需求的解决方案。 首先,让我们理解“串口Socket通讯”。串口通信是一种古老的接口标准,广泛用于设备间的通信...