android 视频解码demo,Android SurfaceView播放视频源码

dba8297b125b676f6a755209347328f3.png

SurfaceView

先来介绍一下大部分软件如何解析一段视频流。首先它需要先确定视频的格式,这个和解码相关, 不同的格式视频编码不同,不是这里的重点。知道了视频的编码格式后,再通过编码格式进行解码,最后得到一帧一帧的图像,并把这些图像快速的显示在界面上, 即为播放一段视频。SurfaceView在Android中就是完成这个功能的。

既然SurfaceView是配合MediaPlayer使用的,MediaPlayer也提供了相应的方法设置SurfaceView显示图片,只需要为MediaPlayer指定SurfaceView显示图像即可。它的完整签名如下:

void setDisplay(SurfaceHolder sh)

它需要传递一个SurfaceHolder对象,SurfaceHolder可以理解为SurfaceView装载需要显示的一帧帧图像的容器,它可以通过SurfaceHolder.getHolder()方法获得。

使用MediaPlayer配合SurfaceView播放视频的步骤与播放使用MediaPlayer播放MP3大体一致,只需要额外设置显示的SurfaceView即可。

SurfaceView双缓冲

上面有提到,SurfaceView和大部分视频应用一样,把视频流解析成一帧帧的图像进行 显示,但是如果把这个解析的过程放到一个线程中完成,可能在上一帧图像已经显示过后,下一帧图像还没有来得及解析,这样会导致画面的不流畅或者声音和视频 不同步的问题。所以SurfaceView和大部分视频应用一样,通过双缓冲的机制来显示帧图像。那么什么是双缓冲呢?双缓冲可以理解为有两个线程轮番去 解析视频流的帧图像,当一个线程解析完帧图像后,把图像渲染到界面中,同时另一线程开始解析下一帧图像,使得两个线程轮番配合去解析视频流,以达到流畅播 放的效果。

SurfaceHolder

SurfaceView内部实现了双缓冲的机制,但是实现这个功能是非常消耗系统内存的。因为移动设备的局限性,Android在设计的时候规 定,SurfaceView如果为用户可见的时候,创建SurfaceView的SurfaceHolder用于显示视频流解析的帧图片,如果发现 SurfaceView变为用户不可见的时候,则立即销毁SurfaceView的SurfaceHolder,以达到节约系统资源的目的。

如果开发人员不对SurfaceHolder进行维护,会出现最小化程序后,再打开应用的时候,视频的声音在继续播放,但是不显示画面了的情况,这 就是因为当SurfaceView不被用户可见的时候,之前的SurfaceHolder已经被销毁了,再次进入的时候,界面上的 SurfaceHolder已经是新的SurfaceHolder了。所以SurfaceHolder需要我们开发人员去编码维护,维护 SurfaceHolder需要用到它的一个回调,SurfaceHolder.Callback(),它需要实现三个如下三个方法:

void surfaceDestroyed(SurfaceHolder holder):当SurfaceHolder被销毁的时候回调。

void surfaceCreated(SurfaceHolder holder):当SurfaceHolder被创建的时候回调。

void surfaceChange(SurfaceHolder holder):当SurfaceHolder的尺寸发生变化的时候被回调。

以下是这三个方法的调用的过程,在应用中分别为SurfaceHolder实现了这三个方法,先进入应用,SurfaceHolder被创建,创建 好之后会改变SurfaceHolder的大小,然后按Home键回退到桌面销毁SurfaceHolder,最后再进入应用,重新 SurfaceHolder并改变其大小。

SurfaceView的Demo示例

上面讲了那么多关于SurfaceView的内容,下面通过一个Demo简单演示一下 SurfaceView如何播放视频,加了一个滚动条,用于显示进度,还可以拖动滚动条选择播放位置,Demo的注释比较完整,这里不再累述,视频是在网 上随便找的,朋友们运行的时候保证/sdcard/ykzzldx.mp4,这个目录下有这个文件。

布局文件:activity_main.xml

4d6a72478d6d201b0d571bf26cb25edd.png

实现代码:

packagecn.bgxt.surfaceviewdemo;

importjava.io.File;

importandroid.media.AudioManager;

importandroid.media.MediaPlayer;

importandroid.media.MediaPlayer.OnCompletionListener;

importandroid.media.MediaPlayer.OnErrorListener;

importandroid.media.MediaPlayer.OnPreparedListener;

importandroid.os.Bundle;

importandroid.app.Activity;

importandroid.util.Log;

importandroid.view.SurfaceHolder;

importandroid.view.SurfaceHolder.Callback;

importandroid.view.SurfaceView;

importandroid.view.View;

importandroid.widget.Button;

importandroid.widget.EditText;

importandroid.widget.SeekBar;

importandroid.widget.SeekBar.OnSeekBarChangeListener;

importandroid.widget.Toast;

publicclassMainActivityextendsActivity {

privatefinalString TAG ="main";

privateEditText et_path;

privateSurfaceView sv;

privateButton btn_play, btn_pause, btn_replay, btn_stop;

privateMediaPlayer mediaPlayer;

privateSeekBar seekBar;

privateintcurrentPosition =0;

privatebooleanisPlaying;

@Override

protectedvoidonCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

seekBar = (SeekBar) findViewById(R.id.seekBar);

sv = (SurfaceView) findViewById(R.id.sv);

et_path = (EditText) findViewById(R.id.et_path);

btn_play = (Button) findViewById(R.id.btn_play);

btn_pause = (Button) findViewById(R.id.btn_pause);

btn_replay = (Button) findViewById(R.id.btn_replay);

btn_stop = (Button) findViewById(R.id.btn_stop);

btn_play.setOnClickListener(click);

btn_pause.setOnClickListener(click);

btn_replay.setOnClickListener(click);

btn_stop.setOnClickListener(click);

// 为SurfaceHolder添加回调

sv.getHolder().addCallback(callback);

// 4.0版本之下需要设置的属性

// 设置Surface不维护自己的缓冲区,而是等待屏幕的渲染引擎将内容推送到界面

// sv.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);

// 为进度条添加进度更改事件

seekBar.setOnSeekBarChangeListener(change);

}

privateCallback callback =newCallback() {

// SurfaceHolder被修改的时候回调

@Override

publicvoidsurfaceDestroyed(SurfaceHolder holder) {

Log.i(TAG, "SurfaceHolder 被销毁");

// 销毁SurfaceHolder的时候记录当前的播放位置并停止播放

if(mediaPlayer !=null&& mediaPlayer.isPlaying()) {

currentPosition = mediaPlayer.getCurrentPosition();

mediaPlayer.stop();

}

}

@Override

publicvoidsurfaceCreated(SurfaceHolder holder) {

Log.i(TAG, "SurfaceHolder 被创建");

if(currentPosition >0) {

// 创建SurfaceHolder的时候,如果存在上次播放的位置,则按照上次播放位置进行播放

play(currentPosition);

currentPosition = 0;

}

}

@Override

publicvoidsurfaceChanged(SurfaceHolder holder,intformat,intwidth,

intheight) {

Log.i(TAG, "SurfaceHolder 大小被改变");

}

};

privateOnSeekBarChangeListener change =newOnSeekBarChangeListener() {

@Override

publicvoidonStopTrackingTouch(SeekBar seekBar) {

// 当进度条停止修改的时候触发

// 取得当前进度条的刻度

intprogress = seekBar.getProgress();

if(mediaPlayer !=null&& mediaPlayer.isPlaying()) {

// 设置当前播放的位置

mediaPlayer.seekTo(progress);

}

}

@Override

publicvoidonStartTrackingTouch(SeekBar seekBar) {

}

@Override

publicvoidonProgressChanged(SeekBar seekBar,intprogress,

booleanfromUser) {

}

};

privateView.OnClickListener click =newView.OnClickListener() {

@Override

publicvoidonClick(View v) {

switch(v.getId()) {

caseR.id.btn_play:

play(0);

break;

caseR.id.btn_pause:

pause();

break;

caseR.id.btn_replay:

replay();

break;

caseR.id.btn_stop:

stop();

break;

default:

break;

}

}

};

/*

* 停止播放

*/

protectedvoidstop() {

if(mediaPlayer !=null&& mediaPlayer.isPlaying()) {

mediaPlayer.stop();

mediaPlayer.release();

mediaPlayer = null;

btn_play.setEnabled(true);

isPlaying = false;

}

}

/**

* 开始播放

*

* @param msec 播放初始位置

*/

protectedvoidplay(finalintmsec) {

// 获取视频文件地址

String path = et_path.getText().toString().trim();

File file = newFile(path);

if(!file.exists()) {

Toast.makeText(this,"视频文件路径错误",0).show();

return;

}

try{

mediaPlayer = newMediaPlayer();

mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);

// 设置播放的视频源

mediaPlayer.setDataSource(file.getAbsolutePath());

// 设置显示视频的SurfaceHolder

mediaPlayer.setDisplay(sv.getHolder());

Log.i(TAG, "开始装载");

mediaPlayer.prepareAsync();

mediaPlayer.setOnPreparedListener(newOnPreparedListener() {

@Override

publicvoidonPrepared(MediaPlayer mp) {

Log.i(TAG, "装载完成");

mediaPlayer.start();

// 按照初始位置播放

mediaPlayer.seekTo(msec);

// 设置进度条的最大进度为视频流的最大播放时长

seekBar.setMax(mediaPlayer.getDuration());

// 开始线程,更新进度条的刻度

newThread() {

@Override

publicvoidrun() {

try{

isPlaying = true;

while(isPlaying) {

intcurrent = mediaPlayer

.getCurrentPosition();

seekBar.setProgress(current);

sleep(500);

}

} catch(Exception e) {

e.printStackTrace();

}

}

}.start();

btn_play.setEnabled(false);

}

});

mediaPlayer.setOnCompletionListener(newOnCompletionListener() {

@Override

publicvoidonCompletion(MediaPlayer mp) {

// 在播放完毕被回调

btn_play.setEnabled(true);

}

});

mediaPlayer.setOnErrorListener(newOnErrorListener() {

@Override

publicbooleanonError(MediaPlayer mp,intwhat,intextra) {

// 发生错误重新播放

play(0);

isPlaying = false;

returnfalse;

}

});

} catch(Exception e) {

e.printStackTrace();

}

}

/**

* 重新开始播放

*/

protectedvoidreplay() {

if(mediaPlayer !=null&& mediaPlayer.isPlaying()) {

mediaPlayer.seekTo(0);

Toast.makeText(this,"重新播放",0).show();

btn_pause.setText("暂停");

return;

}

isPlaying = false;

play(0);

}

/**

* 暂停或继续

*/

protectedvoidpause() {

if(btn_pause.getText().toString().trim().equals("继续")) {

btn_pause.setText("暂停");

mediaPlayer.start();

Toast.makeText(this,"继续播放",0).show();

return;

}

if(mediaPlayer !=null&& mediaPlayer.isPlaying()) {

mediaPlayer.pause();

btn_pause.setText("继续");

Toast.makeText(this,"暂停播放",0).show();

}

}

}

【编辑推荐】

【责任编辑:闫佳明 TEL:(010)68476606】

点赞 0

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值