手把手教你做视频播放器(二)

第2节 获取视频信息

要知道设备上有哪些可以被播放的视频文件,一般来讲有两个方法,

  1. 遍历设备磁盘上所有的目录,根据文件的后缀名,把这些目录中所有的视频文件都找出来;
  2. 向安卓系统提供的Media Provider提出查询请求,从而获取我们希望的视频文件信息;

从“减小开发难度,利用安卓系统自身功能,选择最简单的方案”的角度出发,我们采用Media Provider

2.1 ContentProvider

ContentProvider是安卓系统的四大组件之一,为别的组件(Activity、Service)“提供内容”。它就像是一个拥有某种数据的网站,安卓系统运行的其它组件可以通过“网址”访问这个网站,获取需要查询的数据。

  1. ContentProvider可以是私有的,只能为它所在的应用提供数据访问请求;
  2. ContentProvider也可以是公开的,为别的应用程序提供数据访问请求。

    当我们自己开发一个ContentProvider的时候,就要在应用的AndroidManifest.xml文件中声明它的存在,

<application
        android:allowBackup="true"
        ....../>

        <activity android:name=".MyActivity" />

    <provider
            android:name=".MyContentProvider"
            android:authorities="com.anddle.videoplayer"
            android:exported="false"/>  -->false为私有的,true为公开的

</application>

从这里也可以看出,它和Activity的地位是一样的,所以与Activity一样并称为安卓系统的四大组件之一。

2.2 系统级的ContentProvider

安卓系统上,有一个叫做Media ProviderContentProvider。它作为系统级别的应用程序在系统上运行,专门负责收集多媒体文件(音频、视频、文件)相关的信息。

Media Provider在开机启动后,会在后台“监听”磁盘上文件的变化,特定情况下,会自动更新多媒体文件的信息,例如磁盘上是否增加了媒体文件,是否被删除了媒体文件,有的媒体文件名称是否发生了修改等等。

所以当任何应用想获取这类文件相关的信息时,就可以向Media Provider发起查询的请求。Media Provider帮我们完成了视频文件信息的收集,因此,我们就不用自己去遍历磁盘上的文件进行视频文件的收集和整理了。

除了Media Provider,系统还提供了

  1. Contacts Provider:用来查询联系人信息;
  2. Calendar Provider:用来提供日历相关信息的查询;
  3. Bookmark Provider:用来提供书签信息的查询;

其它的就不再一一列出了。

2.3 使用Media Provider的缺点

使用Media Provider有一点需要注意,Media Provider对磁盘多媒体文件的“监控”,并不是实时的。当删除磁盘上一个已有的视频文件时,Media Provider并不会马上知道,而是要等到下一次的扫描之后,才会更新这个信息。因此,如果使用Media Provider做为我们提供视频信息的来源,就要考虑到“一个视频刚好被修改了,但是还没有来得及在Media Provider中更新信息”的情况。

2.4 Media Provider查询视频文件

  1. 确定向Media Provider发出查询请求的地址-uri,它就像访问网站时,要输入的网址一样。系统提供了两个位置的uri,一个是指向内部存储的uri,一个是指向外部存储的uri。我们要查询的视频文件都是存放在外部存储地址上的;

    Uri uri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
  2. 确定要请求的视频文件信息。在视频列表中,我们需要展示视频的标题、创建时间,还需要播放它时使用的文件所在地址。这些信息在Media Provider中都对应着查询它们使用的字段名称;
    String[] searchKey = new String[] {
    MediaStore.Video.Media.TITLE, -->对应文件的标题
    MediaStore.Images.Media.DATA, -->对应文件的存放位置
    MediaStore.Images.Media.DATE_ADDED -->对应文件的创建时间
    };
  3. 确定查询的条件。我们之前假设过只关心那些叫做Video的目录。因此我们要确定的只是查询到的文件路径中,包含有/Video这个字段。
    String where = MediaStore.Video.Media.DATA + " like \"%"+"/Video"+"%\"";

    这个条件参数的写法就和SQL数据库语言的语法一样。这里我们不打算讲SQL语法,只要知道在我们这个例子中这样使用就好了;

  4. 设定查询结果的排序方式,使用默认的排序方式就可以了,
    String sortOrder = MediaStore.Video.Media.DEFAULT_SORT_ORDER;
  5. 获取ContentResolver对象,让它使用前面的参数向Media Provider发起查询请求;查询的结果存放在Cursor--指标当中;
    ContentResolver resolver = getContentResolver();
    Cursor cursor = resolver.query(
                    uri,
                    searchKey,
                    where,
                    null,
                    sortOrder);
  6. 遍历Cursor,得到它指向的每一条查询到的信息;当Cursor指向某条数据的时候,我们就获取它携带的每个字段的值;
    while(cursor.moveToNext())
    {
        String path = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA));
        String name = cursor.getString(cursor.getColumnIndex(MediaStore.Video.Media.TITLE));
        String createdTime = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATE_ADDED));
        ......
    }
  7. 存放获取的视频文件信息,创建一个VideoItem类
    public class VideoItem {
    
        String name;
        String path;
        Bitmap thumb;
        String createdTime;
    
        VideoItem(String strPath, String strName, String createdTime) {
    
            this.path = strPath;
            this.name = strName;
            ......
       }
    }

    将视频文件创建的时间从Unix格式的时间戳,转换成“年月日分”这种可读的形势,

    VideoItem(String strPath, String strName, String createdTime) {
        ......
        SimpleDateFormat sf = new SimpleDateFormat("yy年MM月dd日HH时mm分");
        Date d = new Date(Long.valueOf(createdTime)*1000);
        this.createdTime = sf.format(d);
    }

    获取视频文件的缩略图,Android SDK提供了一个利用视频文件地址获取视频缩略图的工具,用起来非常简单的,通过它将得到缩略图的Bitmap;

    VideoItem(String strPath, String strName, String createdTime) {
        ......
        this.thumb = ThumbnailUtils.createVideoThumbnail(path, MediaStore.Images.Thumbnails.MINI_KIND);
    }

    图片生成的尺寸可以通过第二个参数设置,MINI_KIND

    表示小的缩略图;FULL_SCREEN_KIND表示大尺寸的缩略图;MICRO_KIND表示超小图的缩略图。这里我们采用的是MINI_KIND

  8. Cursor使用完了之后要把它关闭掉,

    “`java

    cursor.close();

“`

整理一下前面的各个步骤,获取外部存储上Video目录中所有视频文件的方式如下,

Uri uri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;

String[] searchKey = new String[] {
    MediaStore.Video.Media.TITLE,
    MediaStore.Images.Media.DATA,
    MediaStore.Images.Media.DATE_ADDED
};
String [] keywords = null;
String where = MediaStore.Video.Media.DATA + " like \"%"+"/Video"+"%\"";
String sortOrder = MediaStore.Video.Media.DEFAULT_SORT_ORDER;

ContentResolver resolver = getContentResolver();
Cursor cursor = resolver.query(
                    uri,
                    searchKey,
                    where,
                    keywords,
                    sortOrder);

if(cursor != null)
{
    while(cursor.moveToNext())
    {
        String path = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA));
        String name = cursor.getString(cursor.getColumnIndex(MediaStore.Video.Media.TITLE));

        String createdTime = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATE_ADDED));

        VideoItem item = new VideoItem(path, name, createdTime);
        ......
    }

    cursor.close();
}

最后一点千万不要忘记,要在应用的AndroidManifest.xml文件中,添加读取外部存储器的权限,

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.anddle.anddleplayer">

    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    ......

</manifest>

第3节 异步方式获取视频信息

获取视频信息所需要的时间是个不能确定的事情。如果视频很少,也许几十毫秒就能完成,如果视频很多(比如几十个),也许就要花二十多秒。

安卓应用只有一个主线程-各个组件都是在这个线程中运行。作为组件的之一的Activity就是在这个线程中更新应用界面的,例如,用户点击界面上的一个按钮,按钮得到响应,整个过程就是在这个主线程里。所以这个主线程绝对不可以做耗时的操作。假如在按钮中做了耗时的操作,那么当它进行耗时操作的时候,你去点击界面上的其它按钮是不会有反应的,就好像程序冻在了那里。

我们的代码一旦连续占用这个线程超过一定的时间,系统就会弹出“程序无响应的”提示,这个提示叫做ANR-Applicatin No Response。

因此,我们可以考虑把获取视频信息的操作放到一个单独的线程thread中进行。

这就好比你在正在做一件事情A,突然另一件事情B来打扰你,你不得不停下手头的工作来完成,做完了才能继续之前的工作;这时如果有另外一个人(另一个线程)来帮助你,把事情B全部包揽了,那你就不用分心了。当另一个人把事情B做完后,告诉你一声就可以了。

3.1 异步操作

启动一个新的线程,分担耗时工作的方法是一种异步操作:我让你帮我做一件事情,布置任务后,我就去做其他的事情了,等你做完了再告诉我结果;

与它对应的是同步操作:我让你帮我做一件事情,布置任务后,我啥也不做,就等着你做完了告诉我结果;

获取视频信息是个异步操作,启动一个新线程-工作线程thread-查询视频信息,查询完成后,工作线程再将结果通知到主线程,让主线程将查询到结果的结果显示到界面上。界面的更新一定要在主线程中进行,不能在别的线程修改,否则系统后提示运行错误,这一点相当重要。因此我们一定要将查询的结果发送给主线程,让主线程处理界面的更新。

3.2 异步操作的方案

安卓系统提供的异步操作方案有:

  1. 创建工作线程thread和Handler,利用Handler在工作线程和主线程之间传递数据;
  2. 使用AsyncTask帮助类,AsyncTask中封装了工作线程,通过AsyncTask完成工作线程和主线程之间的数据传递;

这里虽然将AsyncTask看成是一个单独的方案,但实际上它也是通过方案1实现的,只不过对于使用者来讲更加方便而已。

这里我们选择方案2。因为,

  1. 使用场景简单,只是单个任务的异步操作,没有多个线程之间的数据同步考虑;
  2. 使用方便,不用考虑太多的新线程创建的细节;

3.3 AsyncTask的使用

3.3.1 AsyncTask的用法

AsyncTask需要被继承成为一个新的子类来使用,在被继承时,要指定三种参数的类型-Param Progress Result,还需要实现doInBackground(Param...)函数,此外通常还要实现onProgressUpdate(Progress...) onPostExecute(Result) 两个回调函数。

class MyTask  extends AsyncTask<Param, Progress, Result> {

    @Override
    protected Result doInBackground(Param... params) {
        return result;
    }

    @Override
    protected void onProgressUpdate(Progress... progresses) {

    }

    @Override
    protected void onPostExecute(Result result) {

    }

    @Override
    protected void onCancelled() {

    }
}
  1. doInBackground(Param... params)函数:传入参数的Param类型就是AsyncTask<Param, Progress, Result>中指定的Param类型。它运行在新创建的工作线程当中。

    使用MyTask时,要在主线程中使用excute()方法传入不定长参数,让Task运行起来,

    MyTask task = new MyTask();
    task.excute(param0, param1, ..., paramN);

    不定长参数会以数组的形式传递到doInBackground()函数当中,

    @Override
    protected Result doInBackground(Param... params) {
        Param param0 = params[0];
        Param param1 = params[1];
        ......
        Param paramN = params[N];
    
        return result;
    }
  2. onProgressUpdate(Progress... progresses)函数:传入参数的Progress类型就是AsyncTask<Param, Progress, Result>中指定的Progress类型。

    doInBackground()中执行的是一个很耗时的工作,有时需要向主线程报告当前的运行状况,这就要使用到publishProgress()函数,publishProgress()也是使用的不定长参数,

    @Override
    protected Result doInBackground(Param... params) {
        ......
        publishProgress(progress1, progress2, ..., progressN)
    
        return result;
    }

    不定长参数会以数组的形式传递到onProgressUpdate()函数当中,

    @Override
    protected void onProgressUpdate(Progress... progresses) {
        Progress progress0 = progresses[0];
        Progress progress1 = progresses[1];
        ......
        Progress progressN = progresses[N];
    
    }
  3. onPostExecute(Result result)函数:传入参数的Result类型就是AsyncTask<Param, Progress, Result>中指定的Result类型。

    doInBackground()函数返回的类型也是Result

    @Override
    protected Result doInBackground(Param... params) {
        ......
    
        return result;
    }

    返回的结果作为参数传递给onPostExecute()函数,

    @Override
    protected void onPostExecute(Result result) {
    
    }
  4. onCancel()函数会在调用者取消AsyncTask的工作的时候被触发。

    要取消AsyncTask的工作,首先要在主线程中调用cancel()方法,

    task.cancel(true);

    因为在doInBackground()中执行的是一个很耗时的工作,需要时不时的检查自己是否被取消执行了,

    @Override
    protected Result doInBackground(Param... params) {
        ......
        if(isCancelled())
        {
            ......
            return result;
        }
    
        ......
        return result;
    }

    最后,onCancelled()函数会被触发,这个函数会在主线程中被执行,

    @Override
    protected void onCancelled() {
    
    }


综合上面的分析,自定义一个AsyncTask的方法如下,

class MyTask extends AsyncTask<Param, Progress, Result> {

    @Override
    protected Result doInBackground(Param... params) {
        Param param0 = params[0];
        Param param1 = params[1];
        ......
        Param paramN = params[N];

        while(!isCancelled())
        {
            ......
            publishProgress(progress1, progress2, ..., progressN);
        }

        return result;
    }

    @Override
    protected void onProgressUpdate(Progress... progresses) {
        Progress progress0 = progresses[0];
        Progress progress1 = progresses[1];
        ......
        Progress progressN = progresses[N];
        ......
    }

    @Override
    protected void onPostExecute(Result result) {

    }

    @Override
    protected void onCancelled() {

    }
}

使用一个AsyncTask的方法如下,

MyTask task = new MyTask();
task.excute(param0, param1, ..., paramN);
......
task.cancel(true);

3.3.2 获取视频信息的AsyncTask

根据我们的需要,自己定义个AsyncTaskVideoUpdateTask

  1. 不需要为新创建的线程传入参数;所以Param设置成Object
  2. 因为查询的过程很长,所以需要时不时通知主线程查询的状态,每查询到一条,就将视频数据传递给主线程;所以Progress设置成VideoItem
  3. 查询的结果已经在查询的过程中发送给了主线程,全部完成后,不需要再传递什么结果给主线程了,所以Result设置成Void
  4. 将查询视频信息的操作放到doInBackground()中进行,这是一个新创建的工作线程;
  5. 工作线程中,每发现一个视频,就通知给主线程;
class VideoUpdateTask extends AsyncTask<Object, VideoItem, Void> {

    @Override
    protected Void doInBackground(Object... params) {

        Uri uri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;

        String[] searchKey = new String[] {
            MediaStore.Video.Media.TITLE,
            MediaStore.Images.Media.DATA,
            MediaStore.Images.Media.DATE_ADDED
        };
        String [] keywords = null;
        String where = MediaStore.Video.Media.DATA + " like \"%"+"/Video"+"%\"";
        String sortOrder = MediaStore.Video.Media.DEFAULT_SORT_ORDER;

        ContentResolver resolver = getContentResolver();
        Cursor cursor = resolver.query(
                            uri,
                            searchKey,
                            where,
                            keywords,
                            sortOrder);

        if(cursor != null)
        {
            while(cursor.moveToNext() && ! isCancelled())
            {
                String path = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA));
                String name = cursor.getString(cursor.getColumnIndex(MediaStore.Video.Media.TITLE));

                String createdTime = cursor.getString(cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATE_ADDED));

                 VideoItem item = new VideoItem(path, name, createdTime);

                publishProgress(item);
            }

            cursor.close();
        }

        return null;
    }

    @Override
    protected void onProgressUpdate(VideoItem... progresses) {
        VideoItem item = (VideoItem) progresses[0];

        //更新界面
        ......
    }

    @Override
    protected void onPostExecute(Result result) {
        //更新界面
        ......
    }

    @Override
    protected void onCancelled() {
        //更新界面
        ......
    }
}

在视频列表Activity创建的时候,启动VideoUpdateTask,开始查询符合我们要求的视频信息。

private AsyncTask mVideoUpdateTask;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_video_list);

    mVideoUpdateTask = new VideoUpdateTask();
    mVideoUpdateTask.execute();
}

在视频列表Activity退出的时候,判断VideoUpdateTask是否还在运行,如果还在运行,就让它停止,

@Override
protected void onDestroy() {
    super.onDestroy();

    if((mVideoUpdateTask != null) &&
        (mVideoUpdateTask.getStatus() == AsyncTask.Status.RUNNING))
    {
        mVideoUpdateTask.cancel(true);
    }

    mVideoUpdateTask = null;
}

onCreate()onDestroy()是Activity生命周期的一部分,当一个Activity被创建的时候会调用到onCreate(),当Activity被退出销毁的时候会调用到onDestroy()。所以在这两个地方使用VideoUpdateTask是一个合适的选择。

时间: 2024-10-13 05:12:23

手把手教你做视频播放器(二)的相关文章

手把手教你做视频播放器(一)

前言 通过"计算器"应用我们已经熟悉了安卓应用开发的大致流程,具备了开发的初步知识. 接下来,我们将开始制作一个"视频播放器"应用,进一步加深对程序开发的学习. 当完成这个"视频播放器"应用后,大家就能够独立开发一类稍微复杂点的安卓应用了. 本文针对的读者是: 对安卓开发有了初步认识,但还没有什么经验的新人: 对已有的安卓开发经验没有系统化整理的开发者: 在开始以前,假设各位已经做好了如下准备: 一台开发用笔记本电脑,并搭建好了开发环境: 一部安

手把手教你做视频播放器(六)(完)

第8节 横屏的播放界面 在设备旋转成横屏的时候,视频将自动进行全屏播放. 8.1 播放器横屏布局 我们要为全屏播放界面设置一个新的布局,这个布局里面只用来播放视频,不需要显示任何视频信息, <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="m

手把手教你做视频播放器(五)

第7节 竖屏的播放界面 播放视频的功能放在一个单独的Activity当中.我们将为它们设置横竖屏两种布局. 在竖屏的时候,上半部分播放视频,下半部分显示视频信息: 在设备旋转成横屏的时候,视频进行全屏播放: 7.1 启动视频播放界面 当点击视频列表的视频项时,就启动播放器播放对应的视频.这里我们要创建一个名字叫做VideoPlayer的Activity,用它来完成视频播放的任务. 另外,还要为ListView添加一个数据项点击时的监听函数, 实现ListView的OnItemClickListe

手把手教你做视频播放器(五)-视频列表的横屏

第6节 视频列表的横屏 设备在横放或者竖放的时候都会自动调整应用的布局,进行对应的横屏显示或者竖屏显示.我们的视频列表也是如此. 我们可以用两种方法处理设备旋转的问题, 让Activity不跟随设备方向的旋转而旋转,它只有竖屏(或只有横屏)的界面.要做到这一点很容易,在AndroidManifest.xml文件中,给这个Activity组件加上android:screenOrientation="portrait" (保持竖屏)或者android:screenOrientation=&

手把手教你做视频播放器(四)

第5节 刷新与停止刷新列表 虽然经过我们的假设,忽略了很多不需要关注的视频文件,但设备上依然有可能有很多的满足了我们假设条件的视频存在,这时就需要一个"取消刷新"的功能. 如果视频还没有刷新完,就被取消了,然后又希望继续刷新,那么还需要一个手动开始"刷新"的功能. 因此,准备在ActionBar的右上角,设置一个菜单项,让用户可以"刷新",也能"停止刷新". 5.1 添加刷新菜单项 在制作"计算器"的文档里

手把手教你做视频播放器(三)

第4章 展示视频列表 在应用界面当中,经常需要使用列表来展示内容. Android SDK提供了ListView控件,来实现这种效果. ListView需要和Adapter配合使用,ListView负责内容的显示,Adapter负责为ListView提供要展示的数据. 4.1 ListView的使用方法 使用ListView展示内容,通常分下面几个步骤, 在布局文件中设置ListView布局: <ListView xmlns:android="http://schemas.android.

微信测试工程师手把手教你做弱网络模拟测试

微信测试工程师手把手教你做弱网络模拟测试 Posted by 腾讯优测 | 3,152 views 小优有话说: app研发不同于实验室里做研究,哪里有"理想环境". 理想里,用户用着性能卓越的手机,连着畅通无阻的wifi网络. "哇塞!这个app好用到飞起!" 现实是,他们可能正用着你闻所未闻的机型,穿梭于地铁.公交.火车.乡间.大山-.. 信号"若隐若现,扑朔迷离" "我去!又crash了!" "唉,怎么又连不上

手把手教你做关键词匹配项目(搜索引擎)---- 第三天

第三天 小王(运营总监)看到小丁丁整天都在淘宝.百度.魔方.拍拍上面淘关键词,每天花费的时间好长,工作效率又低,拿着这个借口来找到我. 说到:小帅帅,你看小丁丁每天都在淘宝.百度.魔方.拍拍上面淘关键词花费的时间好长,你能不能帮帮忙,看看能不能让系统自己做啦,这样可以节省好多人力,带来的效益多高.(0 其实就是为了掩饰他们懒惰 0) 小帅帅一听到可以带来的效益好高,王总还求着我呢 ,马上 两眼冒着星光,是该好好体现, 解决这个问题就可以体现出我的价值. 小帅帅拍着胸膛保证到:王总,这个小KS啦,

UWP Jenkins + NuGet + MSBuild 手把手教你做自动UWP Build 和 App store包

背景 项目上需要做UWP的自动安装包,在以前的公司接触的是TFS来做自动build. 公司要求用Jenkins来做,别笑话我,之前还真不晓得这个东西. 会的同学请看一下支持错误,不会的同学请先自行脑补,我们一步一步的来. 首先我们准备2个安装包,Jenkins,NuGet 都下载最新的好了. 1. 安装Jenkins,下一步下一步.安装好了会自动浏览器跳转到http://localhost:8080/ 如下图 按照提示去C:\Program Files (x86)\Jenkins\secrets