开发者社区> 范大脚脚> 正文

Android 多媒体

简介:
+关注继续查看

第11章  Android多媒体

通过调用Android的API,可以实现相册,播放器,录音和摄像等功能。这一章需要掌握如下功能:

q      多媒体的ContentProvider的调用

q      Camera

q      AudioRecord和AudioTrack

q      MediaPlayer

  11.1  获取多媒体信息

多媒体信息?在pc中的音乐播放器总是很容易的显示歌手名、歌曲名、专辑名、年代。在Android中应该如何获取这些信息呢?

11.1.1  查看多媒体ContentProvider

前面我们学习了ContentProvider来保存和检索数据,Android为常用的数据类型(如:音视频、图片和联系方式等)提供了大量的ContentProvider,它们被定义在android.provider包下。那么我们如何获取多媒体的ContentProvider呢。

(1)       在Eclipse中添加Android自带的FileExplorer视图:

菜单栏->window->show view->other->Android->FileExplorer

(2)       开启模拟器,在FileExplorer中查看data/data/com.android.providers.media/databases/(如图11.1)

(3)       将external.db文件 pull到pc上,用sqlite工具(可以使用火狐插件SQliteManager)查看:(如图11.2)

(4)       查看表结构:

audio_meta:管理sd卡中的音频资源。(如图11.3)

video:管理sd卡中的视频资源。(如图11.4)

images:管理sd卡中的图片。(如图11.5)

 

 

图11.1  多媒体数据库所在包

 2011021810214710.jpg

图11.2  多媒体数据库表

 2011021810220157.gif

 

图11.3  音频表结构(audio_meta)

 2011021810221187.jpg

图11.4  视频表结构(video)

 2011021810231321.jpg

图11.5  图片表结构(images)

2011021810232561.jpg

可能大家看到这里会不明白为什么讲多媒体先要介绍这些。笔者这里给大家列举一个音乐播放器的需求:

1)      获得音乐文件列表及路径

2)      获得音乐文件属性(歌曲名、歌手名、专辑、年代。。)

3)      手动删除内存卡中音乐文件,如何能同步更新文件列表

如果是没有Android经验,大家可能会这样分析:

       文件路径嘛,我调用file.listFile()就可以得到音乐文件列表。如果内存卡里有很多文件,那么这个将会特别的耗时,如果让用户等上10秒去扫描存储卡,用户很有可能将你的应用卸载掉。

       音乐文件属性一般保存在音乐文件中,有的放在文件头,有的放在文件尾,必须读出该文件相关字节中的内容才可以获取音乐文件信息。OK,有个开源的项目,可以解析MP3文件中的文件信息。但是它同样也是耗时的操作。而且学会调用这个开源项目还需要一周的时间。做一个项目没那么难吧?

       在文件管理器中手动删除或添加一个音乐文件,这个时候如何把它更新到音乐列表中呢,当你选择一首歌播放的时候,很有可能已经被删掉了。总不能让它时刻去调用file.listFile()去维护吧?

       那么我们到底应该怎么做呢?你要始终相信Android是强大的,它早已为我们提供了这些功能,我们只需要调用Android提供的API,就可以解决上述的需求。

       现在我们先看个图,图11.6为模拟器开机时,在log中打印的日志。

 

图11.6  开机时的log

2011021810233727.jpg

大家会看到,系统会调用MediaScanner去扫描Internal和External Volume。原来,在开机时,系统会在后台扫描内存和外存设备,将多媒体数据更新到数据库中。同时也会扫描文件信息,这样,我们不费吹灰之力就解决了问题。

小插曲:之前由于笔者还不知道这个方法,在摸索中让同事去研究解析MP3文件信息,花了他一个星期时间,当他做出来了,我花了十分钟找到了这个方法。真是罪过罪过。

那么我们如何得到多媒体数据呢,请看下面的例子:

/**

      * 读取sd卡中的音乐文件

      * @return

      * @throws Exception

      */

     public static ArrayList<Song> readDataFromSD(Context context,int component){

//       Log.i(TAG, "scanFile");

         Cursor cursor = context.getContentResolver().query(

                   MediaStore.Audio.Media.EXTERNAL_CONTENT_URI,

                   new String[] { MediaStore.Audio.Media._ID, MediaStore.Audio.Media.DISPLAY_NAME, MediaStore.Audio.Media.TITLE, MediaStore.Audio.Media.DURATION, MediaStore.Audio.Media.ARTIST, MediaStore.Audio.Media.ALBUM, MediaStore.Audio.Media.YEAR, MediaStore.Audio.Media.MIME_TYPE, MediaStore.Audio.Media.SIZE, MediaStore.Audio.Media.DATA}

                   , MediaStore.Audio.Media.MIME_TYPE+"=? or " + MediaStore.Audio.Media.MIME_TYPE+"=?", new String[]{"audio/mpeg","audio/x-ms-wma"},null);

         ArrayList<Song> songs = null;

         if (cursor.moveToFirst()) {

              songs = SongInfoUtil.getList(cursor, context, component);

         }

         return songs;

     }

 

 

/**

 * @author stayzhang 封装信息至Song

 */

public class SongInfoUtil {

//   private static final String TAG = "ListUtils";

     public static ArrayList<Song> getList(Cursor cursor, Context mContext, int component) {

         Song song = null;

         do {

//            Log.i(TAG, "getList");

              song = new Song();

              song.setFilename(cursor.getString(1));//文件名

              song.setTitle(cursor.getString(2));//歌曲名

              song.setDuration(cursor.getInt(3));//时长

              song.setSinger(cursor.getString(4));//歌手名

              song.setAlbum(cursor.getString(5));//专辑名

              if (cursor.getString(6) != null) {//年代

                   song.setYear(cursor.getString(6));

              } else {

                   song.setYear("未知");

              }

              if ("audio/mpeg".equals(cursor.getString(7).trim())) {//歌曲格式

                   song.setType("mp3");

              } else if ("audio/x-ms-wma".equals(cursor.getString(7).trim())) {

                   song.setType("wma");

              }

              if (cursor.getString(8) != null) {//文件大小

                   float temp = cursor.getInt(8) / 1024f / 1024f;

                   String sizeStr = (temp + "").substring(0, 4);

                   song.setSize(sizeStr + "M");





              } else {

                   song.setSize("未知");

              }

              if (cursor.getString(9) != null) {//文件路径

                   song.setFileUrl(cursor.getString(9));

              }

         } while (cursor.moveToNext());

         cursor.close();

         return dbService.query(component, null, null);

     }

}

同理,图片还有视频文件也可以这样获得,不再赘述。

这样第一个和第二个需求我们已经解决,那第三个需求呢。

当我们手动的删除或添加多媒体文件到存储卡中时,Android会自动扫描这些文件并将其更新到数据库吗?

答案是不会,那么我们如何将数据实时更新到数据库中呢。还记得logcat中打印出来的MediaScanner吗?我们也可以调用MediaScanner这个类去扫描存储卡。但是不能直接调用这个类,只能以广播的形式通知系统,让系统去扫描存储卡指定的URI,扫描完后,再通过ContentProvider查询数据库。

方法如下:

/**

          * 调用系统api扫描sd卡

          */

         private void scanSdCard() {

                   IntentFilter intentFilter = new IntentFilter(

                                     Intent.ACTION_MEDIA_SCANNER_STARTED);

                   intentFilter.addAction(Intent.ACTION_MEDIA_SCANNER_FINISHED);

                   intentFilter.addDataScheme("file");

                   scanReceiver = new ScanSdFilesReceiver();

                   registerReceiver(scanReceiver, intentFilter);

                   sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED,

                                     Uri.parse("file://" + getExternalStorageDirectory().getAbsolutePath)));

         }

 

         /**

          * @author stayzhang 注册扫描开始/完成的广播

          */

         private class ScanSdFilesReceiver extends BroadcastReceiver {

                   public void onReceive(Context context, Intent intent) {

                            String action = intent.getAction();

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

//当系统开始扫描sd卡时,为了用户体验,可以加上一个等待框

                                     scanHandler.sendEmptyMessage(STARTED);

                            }

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

                            //当系统扫描完毕时,停止显示等待框,并重新查询ContentProvider

                                     scanHandler.sendEmptyMessage(FINISHED);

                            }

                   }

         }

OK,关于多媒体的ContentProvider的知识告一段落,下面就是如何调用Android API来使用这些多媒体文件。

分类: android solve
本文转自wanqi博客园博客,原文链接http://www.cnblogs.com/wanqieddy/archive/2011/07/12/2104077.html:如需转载请自行联系原作者

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
阿里云服务器端口号设置
阿里云服务器初级使用者可能面临的问题之一. 使用tomcat或者其他服务器软件设置端口号后,比如 一些不是默认的, mysql的 3306, mssql的1433,有时候打不开网页, 原因是没有在ecs安全组去设置这个端口号. 解决: 点击ecs下网络和安全下的安全组 在弹出的安全组中,如果没有就新建安全组,然后点击配置规则 最后如上图点击添加...或快速创建.   have fun!  将编程看作是一门艺术,而不单单是个技术。
19980 0
阿里云服务器怎么设置密码?怎么停机?怎么重启服务器?
如果在创建实例时没有设置密码,或者密码丢失,您可以在控制台上重新设置实例的登录密码。本文仅描述如何在 ECS 管理控制台上修改实例登录密码。
23523 0
阿里云ECS云服务器初始化设置教程方法
阿里云ECS云服务器初始化是指将云服务器系统恢复到最初状态的过程,阿里云的服务器初始化是通过更换系统盘来实现的,是免费的,阿里云百科网分享服务器初始化教程: 服务器初始化教程方法 本文的服务器初始化是指将ECS云服务器系统恢复到最初状态,服务器中的数据也会被清空,所以初始化之前一定要先备份好。
16589 0
+关注
3656
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
JS零基础入门教程(上册)
立即下载
性能优化方法论
立即下载
手把手学习日志服务SLS,云启实验室实战指南
立即下载