【Android 多媒体开发】 MediaPlayer 网络视频播放器(二)

简介: 【Android 多媒体开发】 MediaPlayer 网络视频播放器(二)

4. MediaPlayer 播放





(1) 设置音量 和 播放载体



设置音量 :



mediaPlayer.setAudioStreamType(2);    /* 设置播放音量 */


设置播放载体 : 调用 setDisplay() 方法, 传入 SurfaceHolder 对象;


mediaPlayer.setDisplay(surface_holder);  /* 设置播放载体 */


(2) 设置各种监听器



设置错误监听器 : 如果出现错误, 会回调该监听器中的方法, 并提供错误码;



/* 设置 MediaPlayer 错误监听器, 如果出现错误就会回调该方法打印错误代码 */
    mediaPlayer.setOnErrorListener(new OnErrorListener() {
    @Override
    public boolean onError(MediaPlayer arg0, int what, int extra) {
      System.out.println("MediaPlayer 出现错误 what : " + what + " , extra : " + extra);
      return false;
    }
    });


设置缓冲进度监听器 : 缓冲有进展后, 回调该监听器中的方法, 传入缓冲的数据百分比;


/* 设置缓冲进度更新监听器 */
    mediaPlayer.setOnBufferingUpdateListener(new OnBufferingUpdateListener() {
    @Override
    public void onBufferingUpdate(MediaPlayer arg0, int percent) {
      /* 打印缓冲的百分比, 如果缓冲 */
      System.out.println("缓冲了的百分比 : " + percent + " %");
    }
    });


设置播放完毕监听器 : 播放完毕后会回调该监听器中的方法;


/* 设置播放完毕监听器 */
    mediaPlayer.setOnCompletionListener(new OnCompletionListener() {
    @Override
    public void onCompletion(MediaPlayer arg0) {
      System.out.println("播放完毕了");
      status.setText("播放完毕");
    }
    });


设置准备完毕回调监听器 : 准备完毕后会回调该方法;


/* 设置准备完毕监听器 */
    mediaPlayer.setOnPreparedListener(new OnPreparedListener() {
    @Override
    public void onPrepared(MediaPlayer arg0) {
      System.out.println("准备完毕");
      /* 设置播放状态 */
      status.setText("播放中");
    }
    });





二. 代码示例



1. 布局文件代码




<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_margin="20dp"
    tools:context="cn.org.octopus.videodemo.HomeActivity"
    tools:ignore="MergeRootFrame" 
    android:orientation="vertical">
    <AutoCompleteTextView 
        android:id="@+id/url"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:singleLine="true"
        android:ellipsize="end"
        android:completionThreshold="1"
        android:text="http://daily3gp.com/vids/747.3gp"
        android:completionHint="选择下载的视频地址"/>
    <SurfaceView 
        android:id="@+id/surface_view"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="3"/>
    <TextView 
        android:id="@+id/status"
        android:layout_width="match_parent"
        android:gravity="center"
        android:layout_height="0dp"
        android:layout_weight="1"
        android:textSize="20dp"
        android:text="状态"/>
    <LinearLayout 
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1"
        android:gravity="center_vertical"
        android:orientation="horizontal">
        <Button 
            android:id="@+id/play"
            android:layout_width="0dp" 
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:onClick="onClick"
            android:text="播放"/>
        <Button 
            android:id="@+id/pause"
            android:layout_width="0dp" 
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:onClick="onClick"
            android:text="暂停"/>
        <Button 
            android:id="@+id/reset"
            android:layout_width="0dp" 
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:onClick="onClick"
            android:text="重放"/>
        <Button 
            android:id="@+id/stop"
            android:layout_width="0dp" 
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:onClick="onClick"
            android:text="停止"/>
    </LinearLayout>
</LinearLayout>




2. AndroidManifest 代码




<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="cn.org.octopus.videodemo"
    android:versionCode="1"
    android:versionName="1.0" >
    <uses-sdk
        android:minSdkVersion="17"
        android:targetSdkVersion="19" />
    <uses-permission android:name="android.permission.INTERNET"/>
    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="cn.org.octopus.videodemo.HomeActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
</manifest>



3. Activity 代码




package cn.org.octopus.videodemo;
import java.io.IOException;
import android.R.anim;
import android.app.Activity;
import android.content.DialogInterface.OnClickListener;
import android.graphics.PixelFormat;
import android.media.MediaPlayer;
import android.media.MediaPlayer.OnBufferingUpdateListener;
import android.media.MediaPlayer.OnCompletionListener;
import android.media.MediaPlayer.OnErrorListener;
import android.media.MediaPlayer.OnPreparedListener;
import android.os.Bundle;
import android.support.v4.view.ViewPager;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.widget.ArrayAdapter;
import android.widget.AutoCompleteTextView;
import android.widget.Button;
import android.widget.TextView;
/** 
 * 
 * 
 * SurfaceHolder  
 *    1. 简介 : 是 Surface 的控制器, 用于控制 SurfaceView 绘图, 处理画布上的动画, 渲染效果, 大小等;
 *    2. 常用方法 : 
 *    -- abstract void addCallback(SurfaceHolder.Callback callback) : 添加一个 SurfaceHolder.Callback 接口对象, 监听 Surface 的开始结束绘制大小改变事件;
 *    -- abstract Canvas lockCanvas() : 锁定画布, 可以获得 Canvas 对象, 之后就可以在 Canvas 上绘图了;
 * 
 * SurfaceHolder.Callback接口 : 
 *    1. Surface 绘图边界 : 所有的绘图工作都在 Surface 创建之后才能进行, 在 Surface 销毁之前结束;
 *    2. Callback 接口对应的 Surface 边界 : surfaceCreated() 方法在开始绘制时回调, surfaceDestroyed() 在 Surface 销毁前回调;
 *    3. 该接口中的方法 : 
 *    -- surfaceChanged() : 在 Surface 大小改变时回调;
 *    -- surfaceCreated() : 在 Surface 创建时回调;
 *    -- surfaceDestroyed() : 在 Surface 销毁时回调;
 * 
 * @author octopus 
 *
 */
public class HomeActivity extends Activity implements SurfaceHolder.Callback {
  private AutoCompleteTextView url;       /* 地址输入框, 带自动提示功能 */
  private SurfaceView surface_view;       /* 播放视频载体 */
  private TextView status;          /* 显示播放状态 */
  private Button play;          /* 播放按钮 */
  private Button pause;          /* 咱提供按钮 */
  private Button reset;          /* 重放按钮 */
  private Button stop;          /* 停止按钮 */
  private MediaPlayer mediaPlayer;        /* 播放器 */
  private SurfaceHolder surface_holder;       /* Surface 控制器 */
  private boolean isStartPlaying;  /* 是否开始了播放 */
  @Override
  protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_home);
  initViews();
  initData();
  }
  /**
  * 初始化成员变量中的组件变量
  */
  private void initViews() {
  /* 通过 findViewById 获取相关方法 */
  url = (AutoCompleteTextView) findViewById(R.id.url);
  surface_view = (SurfaceView) findViewById(R.id.surface_view);
  status = (TextView) findViewById(R.id.status);
  play = (Button) findViewById(R.id.play);
  pause = (Button) findViewById(R.id.pause);
  reset = (Button) findViewById(R.id.reset);
  stop = (Button) findViewById(R.id.stop);
  /* 设置一个列表适配器 */
  String[] urls = {
    "http://daily3gp.com/vids/747.3gp",
    "http://daily3gp.com/vids/Funny%20women%20cannot%20understand.3gp",
    "http://k.youku.com/player/getFlvPath/sid/9409280845322127f6c57_00/st/flv/fileid/0300020100540024BC9E5C08BD8A98D8200E2B-7950-B9A5-8669-DC283BDCC077?K=3a58dc2cdcc532df261dddec&ctype=12&ev=1&oip=1931322792&token=5696&ep=eyaUE0uFVsYE4CDdij8bYHrkJ3IIXP4J9h%2BFg9JjALshTOi%2FmzqjtJTFS4xCHottelMPGJ%2F5qdDnH0JmYfdKrGgQrUfZPPro%2BPbq5dkgxpgDFG1FAc3Qs1SbRTn3",
    "http://k.youku.com/player/getFlvPath/sid/9409280845322127f6c57_00/st/flv/fileid/030002040053FFB59E433100422C39BAFA46CC-4DED-E928-87B8-91706CDB5FF2?K=645d8478a3aa59052411eb8a&ctype=12&ev=1&oip=1931322792&token=5696&ep=eyaUE0uFVsYE4CDdij8bYHrkJ3IIXP4J9h%2BFg9JmALshS57J6zvYspmzTf5CFv0bcFEFGZmA3aHjbDNnYfQ33BwQqkeqMfro%2BYLr5aRSw5AGFW1Ed7uhtlSbRTn3"
  };
  /* 创建数组适配器 */
  ArrayAdapter<String> adapter = new ArrayAdapter<String>(this, 
    android.R.layout.simple_dropdown_item_1line, urls);
  /* 将适配器设置给 AutoCompleteTextView 组件对象 */
  url.setAdapter(adapter);
  /* 使窗口支持透明度, 把当前 Activity 窗口设置成透明, 设置了该选项就可以使用 setAlpha 等函数设置窗口透明度 */
  getWindow().setFormat(PixelFormat.TRANSPARENT);
  }
  /**
  * 初始化相关数据变量
  */
  private void initData() {
  /* 获取并设置 SurfaceHolder 对象 */
  surface_holder = surface_view.getHolder();      /* 根据 SurfaceView 组件, 获取 SurfaceHolder 对象 */
  surface_holder.addCallback(this);         /* 为 SurfaceHolder 设置回调函数, 即 SurfaceHolder.Callback 子类对象 */
  surface_holder.setFixedSize(160, 128);        /* 设置视频大小比例 */
  surface_holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);/* 设置视频类型 */
  }
  /**
  * 设置点击事件
  * @param view
  */
  public void onClick(View view) {
  int id = view.getId();
  switch (id) {
  case R.id.play:
    /* 播放视频直接从 AutoCompleteTextView 中获取字符串, 播放该 url 代表的网络视频 */
    playVideo(url.getText().toString());
    break;
  case R.id.pause:
    if(mediaPlayer != null){
    mediaPlayer.pause();
    status.setText("暂停");
    }
    break;
  case R.id.reset:
    if(mediaPlayer != null){
    mediaPlayer.seekTo(0);
    mediaPlayer.start();
    status.setText("播放中");
    }
    break;
  case R.id.stop:
    if(mediaPlayer != null){
    mediaPlayer.stop();
    mediaPlayer.release();
    isStartPlaying = false;
    status.setText("停止");
    }
    break;
  default:
    break;
  }
  }
  /**
  * 播放网络视频
  * a. 创建并配置 MediaPlayer 对象 (音量, SurfaceHolder)
  * b. 为 MediaPlayer 设置错误监听器, 缓冲进度监听器, 播放完毕监听器, 准备完毕监听器
  * c. 未 MediaPlayer 设置数据源
  * d. 调用 prepare() 进入 Prapared 状态
  * e. 调用 start() 进入 Started 状态
  * 
  * @param dataSource 播放视频的网络地址
  */
  private void playVideo(final String dataSource) {
  /* 点击播放有两种情况 
   * a. 第一次点击 : 需要初始化 MediaPlayer 对象, 设置监听器
   * b. 第二次点击 : 只需要 调用 mediaPlayer 的 start() 方法
   * 两种情况通过 isStartPlaying 点击时间判断 */
  if(isStartPlaying){        /* 如果已经开始了播放, 就直接开始播放 */
    mediaPlayer.start();
  }else{            /* 如果是第一次开始播放, 需要初始化 MediaPlayer 设置监听器等操作 */
    mediaPlayer = new MediaPlayer();    /* 创建 MediaPlayer 对象 */
    mediaPlayer.setAudioStreamType(2);    /* 设置播放音量 */
    mediaPlayer.setDisplay(surface_holder);  /* 设置播放载体 */ 
    /* 设置 MediaPlayer 错误监听器, 如果出现错误就会回调该方法打印错误代码 */
    mediaPlayer.setOnErrorListener(new OnErrorListener() {
    @Override
    public boolean onError(MediaPlayer arg0, int what, int extra) {
      System.out.println("MediaPlayer 出现错误 what : " + what + " , extra : " + extra);
      return false;
    }
    });
    /* 设置缓冲进度更新监听器 */
    mediaPlayer.setOnBufferingUpdateListener(new OnBufferingUpdateListener() {
    @Override
    public void onBufferingUpdate(MediaPlayer arg0, int percent) {
      /* 打印缓冲的百分比, 如果缓冲 */
      System.out.println("缓冲了的百分比 : " + percent + " %");
    }
    });
    /* 设置播放完毕监听器 */
    mediaPlayer.setOnCompletionListener(new OnCompletionListener() {
    @Override
    public void onCompletion(MediaPlayer arg0) {
      System.out.println("播放完毕了");
      status.setText("播放完毕");
    }
    });
    /* 设置准备完毕监听器 */
    mediaPlayer.setOnPreparedListener(new OnPreparedListener() {
    @Override
    public void onPrepared(MediaPlayer arg0) {
      System.out.println("准备完毕");
      /* 设置播放状态 */
      status.setText("播放中");
    }
    });
    new Thread(){
    public void run() {
      try {
      System.out.println("设置数据源");
      mediaPlayer.setDataSource(dataSource);
      mediaPlayer.prepare();
      /* 打印播放视频的时长 */
      System.out.println("视频播放长度 : " + mediaPlayer.getDuration());
      mediaPlayer.start();
      } catch (IllegalStateException e) {
      e.printStackTrace();
      } catch (IOException e) {
      e.printStackTrace();
      }
    };
    }.start();
    /* 设置 MediaPlayer 开始播放标识为 true */
    isStartPlaying = true;
    /* 设置播放状态 */
    status.setText("正在缓冲");
  }
  }
  /**
  * 在 Surface 大小发生改变的时候回调
  * 实现的 SurfaceHolder.Callback 接口方法
  */
  @Override
  public void surfaceChanged(SurfaceHolder arg0, int arg1, int arg2, int arg3) {
  System.out.println("SurfaceHolder.Callback.surfaceChanged : Surface 大小发生改变");
  }
  /**
  * 在 Surface 创建的时候回调, 一般在该方法中开始绘图
  * 实现的 SurfaceHolder.Callback 接口方法
  */
  @Override
  public void surfaceCreated(SurfaceHolder arg0) {
  System.out.println("SurfaceHolder.Callback.surfaceCreated : Surface 开始创建");
  }
  /**
  * 在 Surface 销毁之前回调, 在该方法中停止渲染线程, 释放相关资源
  * 实现的 SurfaceHolder.Callback 接口方法
  */
  @Override
  public void surfaceDestroyed(SurfaceHolder arg0) {
  System.out.println("SurfaceHolder.Callback.surfaceDestroyed : Surface 销毁");
  }
  @Override
  protected void onDestroy() {
  if(mediaPlayer != null)
    mediaPlayer.release();
  super.onDestroy();
  }
}



4. 运行示例


image.png





日志信息 :


octopus@octopus:~/develop/adt-bundle-linux/sdk/tools$ adb logcat -s System.out
--------- beginning of /dev/log/main
--------- beginning of /dev/log/system
I/System.out(21129): 设置数据源
I/System.out(21129): 视频播放长度 : 31200
I/System.out(21129): 准备完毕
I/System.out(21129): 缓冲了的百分比 : 0 %
I/System.out(21129): 缓冲了的百分比 : 37 %
I/System.out(21129): 缓冲了的百分比 : 37 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 播放完毕了
I/System.out(21129): 缓冲了的百分比 : 100 %
目录
相关文章
|
26天前
|
Linux 开发工具 Android开发
FFmpeg开发笔记(六十)使用国产的ijkplayer播放器观看网络视频
ijkplayer是由Bilibili基于FFmpeg3.4研发并开源的播放器,适用于Android和iOS,支持本地视频及网络流媒体播放。本文详细介绍如何在新版Android Studio中导入并使用ijkplayer库,包括Gradle版本及配置更新、导入编译好的so文件以及添加直播链接播放代码等步骤,帮助开发者顺利进行App调试与开发。更多FFmpeg开发知识可参考《FFmpeg开发实战:从零基础到短视频上线》。
101 2
FFmpeg开发笔记(六十)使用国产的ijkplayer播放器观看网络视频
|
3天前
|
API
鸿蒙开发:切换至基于rcp的网络请求
本文的内容主要是把之前基于http封装的库,修改为当前的Remote Communication Kit(远场通信服务),无非就是通信的方式变了,其他都大差不差。
鸿蒙开发:切换至基于rcp的网络请求
|
1月前
|
XML 开发工具 Android开发
FFmpeg开发笔记(五十六)使用Media3的Exoplayer播放网络视频
ExoPlayer最初是为了解决Android早期MediaPlayer控件对网络视频兼容性差的问题而推出的。现在,Android官方已将其升级并纳入Jetpack的Media3库,使其成为音视频操作的统一引擎。新版ExoPlayer支持多种协议,解决了设备和系统碎片化问题,可在整个Android生态中一致运行。通过修改`build.gradle`文件、布局文件及Activity代码,并添加必要的权限,即可集成并使用ExoPlayer进行网络视频播放。具体步骤包括引入依赖库、配置播放界面、编写播放逻辑以及添加互联网访问权限。
130 1
FFmpeg开发笔记(五十六)使用Media3的Exoplayer播放网络视频
|
22天前
|
网络协议 Shell 网络安全
解决两个 Android 模拟器之间无法网络通信的问题
让同一个 PC 上运行的两个 Android 模拟器之间能相互通信,出(qiong)差(ren)的智慧。
22 3
|
22天前
|
Java 程序员 开发工具
Android|修复阿里云播放器下载不回调的问题
虽然 GC 带来了很多便利,但在实际编码时,我们也需要注意对象的生命周期管理,该存活的存活,该释放的释放,避免因为 GC 导致的问题。
28 2
|
3月前
|
安全 网络安全 Android开发
安卓与iOS开发:选择的艺术网络安全与信息安全:漏洞、加密与意识的交织
【8月更文挑战第20天】在数字时代,安卓和iOS两大平台如同两座巍峨的山峰,分别占据着移动互联网的半壁江山。它们各自拥有独特的魅力和优势,吸引着无数开发者投身其中。本文将探讨这两个平台的特点、优势以及它们在移动应用开发中的地位,帮助读者更好地理解这两个平台的差异,并为那些正在面临选择的开发者提供一些启示。
127 56
|
3月前
|
C++
C++ Qt开发:QUdpSocket网络通信组件
QUdpSocket是Qt网络编程中一个非常有用的组件,它提供了在UDP协议下进行数据发送和接收的能力。通过简单的方法和信号,可以轻松实现基于UDP的网络通信。不过,需要注意的是,UDP协议本身不保证数据的可靠传输,因此在使用QUdpSocket时,可能需要在应用层实现一些机制来保证数据的完整性和顺序,或者选择在适用的场景下使用UDP协议。
151 2
|
3月前
|
小程序 数据安全/隐私保护
Taro@3.x+Vue@3.x+TS开发微信小程序,网络请求封装
在 `src/http` 目录下创建 `request.ts` 文件,并配置 Taro 的网络请求方法 `Taro.request`,支持多种 HTTP 方法并处理数据加密。
122 0
Taro@3.x+Vue@3.x+TS开发微信小程序,网络请求封装
|
3月前
|
Android开发
Android 利用MediaPlayer实现音乐播放
本文提供了一个简单的Android MediaPlayer音乐播放示例,包括创建PlayerActivity、配置AndroidManifest.xml和activity_player.xml布局,以及实现播放和暂停功能的代码。
26 0
Android 利用MediaPlayer实现音乐播放
|
3月前
|
数据采集 存储 前端开发
豆瓣评分9.0!Python3网络爬虫开发实战,堪称教学典范!
今天我们所处的时代是信息化时代,是数据驱动的人工智能时代。在人工智能、物联网时代,万物互联和物理世界的全面数字化使得人工智能可以基于这些数据产生优质的决策,从而对人类的生产生活产生巨大价值。 在这个以数据驱动为特征的时代,数据是最基础的。数据既可以通过研发产品获得,也可以通过爬虫采集公开数据获得,因此爬虫技术在这个快速发展的时代就显得尤为重要,高端爬虫人才的收人也在逐年提高。