文章目录
前言
Android多媒体中的——MediaPlayer,我们可以通过这个API来播放音频和视频 该类是Androd多媒体框架中的一个重要组件,通过该类,我们可以以最少的步骤来获取,解码 和播放音视频。它支持三种不同的媒体来源:
- 本地资源
- 内部的URI,比如你可以通过ContentResolver来获取
- 外部URL(流) 对于Android所支持的的媒体格式列表
一、MedaiPlayer状态图
音频/视频文件和流的播放控制作为状态机进行管理。下图显示了受支持的播放控制操作驱动的 MediaPlayer 对象的生命周期和状态。椭圆表示 MediaPlayer 对象可能驻留的状态。弧表示驱动对象状态转换的播放控制操作。有两种类型的弧。带单箭头的弧表示同步方法调用,带双箭头的弧表示异步方法调用。
从这个状态图中,可以出MediaPlayer对象具有以下状态:
-
MediaPlayer在使用new或者reset()之后,它处于idle状态,并且在release()被调用之后,它处于结束状态。
当处于除End状态外,都可以调用reset()把Mediaplayer置为idle状态。
当MediaPlayer对象不再使用后,需要调用relase()释放资源,使它转为End状态。 -
通过注册的setOnErrorListener(android.media.MediaPlayer.OnErrorListener)事件可以获取到一些编程错误信息。
-
idle状态状态下调用setDataSource(),将idle状态的MediaPlayer对象转到initialized状态。(在其他状态调用setDataSource会抛出IllegalStateException错误。)
-
MediaPlayer需要进入Prepared状态后,才能开始播放。
可以通过两种方式到达Prepared状态
1、同步,调用prepare(),调用返回后将对象转移到Prepared状态。
2、异步,调用prepareAsync(),首先在调用返回后(几乎立即发生)进入Preparing状态,而内部播放器引擎继续处理其余的准备工作,直到准备工作完成。之后会触发setOnPreparedListener回调。 -
要开始播放,start()必须调用,调用start()成功后进入started状态。
-
调用pause(),进入Paused状态。
-
调用stop(),进入Stopped状态,stop状态后需要进入Prepared状态才能开始播放。
-
当播放到达流的末尾时,播放完成。
只有当setLooping为false,才会进入PlaybackCompleted。
当setLooping为true,MediaPlayer 对象会会保持在Started状态。 -
在PlaybackCompleted 状态下,调用start()可以从音频/视频源的开头重新开始播放。
二、案例
package com.cs.testmediaplayer;
import android.content.Context;
import android.media.MediaPlayer;
import android.os.Bundle;
import android.util.Log;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.view.WindowManager;
import androidx.appcompat.app.AppCompatActivity;
public class MainActivity extends AppCompatActivity {
SurfaceView mSvVideoPlayer;
private MediaPlayer mMediaPlayer;
private int mPosition = 0;
private boolean hasActiveHolder = false;
private static final String TAG = "MainActivity";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mSvVideoPlayer = findViewById(R.id.sv_video_player);
playVideo();
}
/**
* 播放视频
*/
public void playVideo() {
if (mMediaPlayer == null) {
//实例化MediaPlayer对象
mMediaPlayer = new MediaPlayer();
mSvVideoPlayer.setVisibility(View.VISIBLE);
// mMediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
boolean mHardwareDecoder = false;
// 不维持自身缓冲区,直接显示
// if (Build.VERSION.SDK_INT < Build.VERSION_CODES.HONEYCOMB && mHardwareDecoder) {
mSvVideoPlayer.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
// }
mSvVideoPlayer.getHolder().setFixedSize(getScreenWidth(), getScreenHeight());
mSvVideoPlayer.getHolder().setKeepScreenOn(true);//保持屏幕常亮
mSvVideoPlayer.getHolder().addCallback(new SurFaceCallback());
}
}
/**
* 向player中设置dispay,也就是SurfaceHolder。但此时有可能SurfaceView还没有创建成功,所以需要监听SurfaceView的创建事件
*/
private final class SurFaceCallback implements SurfaceHolder.Callback {
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
}
@Override
public void surfaceCreated(SurfaceHolder holder) {
if (mMediaPlayer == null) {
return;
}
if (!hasActiveHolder) {
play(mPosition);
hasActiveHolder = true;
}
if (mPosition > 0) {
play(mPosition);
mPosition = 0;
}
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
if (mMediaPlayer == null) {
return;
}
mMediaPlayer.setScreenOnWhilePlaying(true);
if (mMediaPlayer.isPlaying()) {
mMediaPlayer.stop();
mPosition = mMediaPlayer.getCurrentPosition();
}
}
private void play(int position) {
try {
//添加播放视频的路径与配置MediaPlayer
// AssetFileDescriptor fileDescriptor = getResources().openRawResourceFd(R.raw.info);
mMediaPlayer.reset();
//给mMediaPlayer添加预览的SurfaceHolder,将播放器和SurfaceView关联起来
mMediaPlayer.setDisplay(mSvVideoPlayer.getHolder());
// mMediaPlayer.setDataSource(path );
//mMediaPlayer.setDataSource("https://vjs.zencdn.net/v/oceans.mp4");
//mMediaPlayer.setDataSource("http://172.29.163.154:8080/resoure/boon.mp4");
//https://storage.googleapis.com/exoplayer-test-media-1/mkv/android-screens-lavf-56.36.100-aac-avc-main-1280x720.mkv
// mMediaPlayer.setDataSource("https://html5demos.com/assets/dizzy.mp4");
mMediaPlayer.setDataSource("https://storage.googleapis.com/exoplayer-test-media-1/mkv/android-screens-lavf-56.36.100-aac-avc-main-1280x720.mkv");
Log.d(TAG, "setDataSource: ");
mMediaPlayer.stop();
// 缓冲
mMediaPlayer.prepareAsync();
mMediaPlayer.setOnBufferingUpdateListener(new BufferingUpdateListener());
mMediaPlayer.setOnPreparedListener(new PreparedListener(position));
mMediaPlayer.setOnCompletionListener(new CompletionListener());
} catch (Exception e) {
e.printStackTrace();
}
}
}
/**
* 缓冲变化时回调
*/
private final class BufferingUpdateListener implements MediaPlayer.OnBufferingUpdateListener {
@Override
public void onBufferingUpdate(MediaPlayer mp, int percent) {
if (mMediaPlayer != null) {
if (mMediaPlayer.isPlaying()) {
mMediaPlayer.pause();
mPosition = mMediaPlayer.getCurrentPosition();
} else {
mMediaPlayer.start();
mMediaPlayer.seekTo(mPosition);
}
}
//mMediaPlayer.stop();
}
}
/**
* 准备完成回调
* 只有当播放器准备好了之后才能够播放,所以播放的出发只能在触发了prepare之后
*/
private final class PreparedListener implements MediaPlayer.OnPreparedListener {
private int position;
public PreparedListener(int position) {
Log.d(TAG, "PreparedListener ");
this.position = position;
mMediaPlayer.start();
}
@Override
public void onPrepared(MediaPlayer mp) {
Log.d(TAG, "onPrepared:start ");
mMediaPlayer.start();
if (position > 0) {
mMediaPlayer.seekTo(position);
}
}
}
/**
* 播放结束时回调
*/
private final class CompletionListener implements MediaPlayer.OnCompletionListener {
@Override
public void onCompletion(MediaPlayer mp) {
/// mMediaPlayer.start();
}
}
@Override
public void onDestroy() {
//释放内存,MediaPlayer底层是运行C++的函数方法,不使用后必需释放内存
if (mMediaPlayer != null) {
if (mMediaPlayer.isPlaying()) {
mMediaPlayer.pause();
mPosition = mMediaPlayer.getCurrentPosition();
}
mMediaPlayer.release();
mMediaPlayer = null;
}
super.onDestroy();
}
private int getScreenWidth() {
return ((WindowManager) getSystemService(Context.WINDOW_SERVICE)).getDefaultDisplay().getWidth();
}
private int getScreenHeight() {
return ((WindowManager) getSystemService(Context.WINDOW_SERVICE)).getDefaultDisplay()
.getHeight();
}
}
注意:如果使用http 明文网络流行资源,需要在AndroidManifest.xml 的application 下添加android:usesCleartextTraffic=“true” , 指示应用程序是否打算使用明文网络流量。目标API级别为27或更低的应用程序的默认值为“ true”。面向API级别28或更高级别的应用默认为“ false”。
同时需要网络权限 '<uses-permission android:name=“android.permission.INTERNET”/>