Android SurfaceView播放視頻源碼
SurfaceView
先來介紹一下大部分軟件如何解析一段視頻流。首先它需要先確定視頻的格式,這個(gè)和解碼相關(guān), 不同的格式視頻編碼不同,不是這里的重點(diǎn)。知道了視頻的編碼格式后,再通過編碼格式進(jìn)行解碼,最后得到一幀一幀的圖像,并把這些圖像快速的顯示在界面上, 即為播放一段視頻。SurfaceView在Android中就是完成這個(gè)功能的。
既然SurfaceView是配合MediaPlayer使用的,MediaPlayer也提供了相應(yīng)的方法設(shè)置SurfaceView顯示圖片,只需要為MediaPlayer指定SurfaceView顯示圖像即可。它的完整簽名如下:
void setDisplay(SurfaceHolder sh)
它需要傳遞一個(gè)SurfaceHolder對(duì)象,SurfaceHolder可以理解為SurfaceView裝載需要顯示的一幀幀圖像的容器,它可以通過SurfaceHolder.getHolder()方法獲得。
使用MediaPlayer配合SurfaceView播放視頻的步驟與播放使用MediaPlayer播放MP3大體一致,只需要額外設(shè)置顯示的SurfaceView即可。
SurfaceView雙緩沖
上面有提到,SurfaceView和大部分視頻應(yīng)用一樣,把視頻流解析成一幀幀的圖像進(jìn)行 顯示,但是如果把這個(gè)解析的過程放到一個(gè)線程中完成,可能在上一幀圖像已經(jīng)顯示過后,下一幀圖像還沒有來得及解析,這樣會(huì)導(dǎo)致畫面的不流暢或者聲音和視頻 不同步的問題。所以SurfaceView和大部分視頻應(yīng)用一樣,通過雙緩沖的機(jī)制來顯示幀圖像。那么什么是雙緩沖呢?雙緩沖可以理解為有兩個(gè)線程輪番去 解析視頻流的幀圖像,當(dāng)一個(gè)線程解析完幀圖像后,把圖像渲染到界面中,同時(shí)另一線程開始解析下一幀圖像,使得兩個(gè)線程輪番配合去解析視頻流,以達(dá)到流暢播 放的效果。
SurfaceHolder
SurfaceView內(nèi)部實(shí)現(xiàn)了雙緩沖的機(jī)制,但是實(shí)現(xiàn)這個(gè)功能是非常消耗系統(tǒng)內(nèi)存的。因?yàn)橐苿?dòng)設(shè)備的局限性,Android在設(shè)計(jì)的時(shí)候規(guī) 定,SurfaceView如果為用戶可見的時(shí)候,創(chuàng)建SurfaceView的SurfaceHolder用于顯示視頻流解析的幀圖片,如果發(fā)現(xiàn) SurfaceView變?yōu)橛脩舨豢梢姷臅r(shí)候,則立即銷毀SurfaceView的SurfaceHolder,以達(dá)到節(jié)約系統(tǒng)資源的目的。
如果開發(fā)人員不對(duì)SurfaceHolder進(jìn)行維護(hù),會(huì)出現(xiàn)最小化程序后,再打開應(yīng)用的時(shí)候,視頻的聲音在繼續(xù)播放,但是不顯示畫面了的情況,這 就是因?yàn)楫?dāng)SurfaceView不被用戶可見的時(shí)候,之前的SurfaceHolder已經(jīng)被銷毀了,再次進(jìn)入的時(shí)候,界面上的 SurfaceHolder已經(jīng)是新的SurfaceHolder了。所以SurfaceHolder需要我們開發(fā)人員去編碼維護(hù),維護(hù) SurfaceHolder需要用到它的一個(gè)回調(diào),SurfaceHolder.Callback(),它需要實(shí)現(xiàn)三個(gè)如下三個(gè)方法:
- void surfaceDestroyed(SurfaceHolder holder):當(dāng)SurfaceHolder被銷毀的時(shí)候回調(diào)。
- void surfaceCreated(SurfaceHolder holder):當(dāng)SurfaceHolder被創(chuàng)建的時(shí)候回調(diào)。
- void surfaceChange(SurfaceHolder holder):當(dāng)SurfaceHolder的尺寸發(fā)生變化的時(shí)候被回調(diào)。
以下是這三個(gè)方法的調(diào)用的過程,在應(yīng)用中分別為SurfaceHolder實(shí)現(xiàn)了這三個(gè)方法,先進(jìn)入應(yīng)用,SurfaceHolder被創(chuàng)建,創(chuàng)建 好之后會(huì)改變SurfaceHolder的大小,然后按Home鍵回退到桌面銷毀SurfaceHolder,最后再進(jìn)入應(yīng)用,重新 SurfaceHolder并改變其大小。
SurfaceView的Demo示例
上面講了那么多關(guān)于SurfaceView的內(nèi)容,下面通過一個(gè)Demo簡(jiǎn)單演示一下 SurfaceView如何播放視頻,加了一個(gè)滾動(dòng)條,用于顯示進(jìn)度,還可以拖動(dòng)滾動(dòng)條選擇播放位置,Demo的注釋比較完整,這里不再累述,視頻是在網(wǎng) 上隨便找的,朋友們運(yùn)行的時(shí)候保證/sdcard/ykzzldx.mp4,這個(gè)目錄下有這個(gè)文件。
布局文件:activity_main.xml
實(shí)現(xiàn)代碼:
- package cn.bgxt.surfaceviewdemo;
- import java.io.File;
- import android.media.AudioManager;
- import android.media.MediaPlayer;
- import android.media.MediaPlayer.OnCompletionListener;
- import android.media.MediaPlayer.OnErrorListener;
- import android.media.MediaPlayer.OnPreparedListener;
- import android.os.Bundle;
- import android.app.Activity;
- import android.util.Log;
- import android.view.SurfaceHolder;
- import android.view.SurfaceHolder.Callback;
- import android.view.SurfaceView;
- import android.view.View;
- import android.widget.Button;
- import android.widget.EditText;
- import android.widget.SeekBar;
- import android.widget.SeekBar.OnSeekBarChangeListener;
- import android.widget.Toast;
- public class MainActivity extends Activity {
- private final String TAG = "main";
- private EditText et_path;
- private SurfaceView sv;
- private Button btn_play, btn_pause, btn_replay, btn_stop;
- private MediaPlayer mediaPlayer;
- private SeekBar seekBar;
- private int currentPosition = 0;
- private boolean isPlaying;
- @Override
- protected void onCreate(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添加回調(diào)
- sv.getHolder().addCallback(callback);
- // 4.0版本之下需要設(shè)置的屬性
- // 設(shè)置Surface不維護(hù)自己的緩沖區(qū),而是等待屏幕的渲染引擎將內(nèi)容推送到界面
- // sv.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
- // 為進(jìn)度條添加進(jìn)度更改事件
- seekBar.setOnSeekBarChangeListener(change);
- }
- private Callback callback = new Callback() {
- // SurfaceHolder被修改的時(shí)候回調(diào)
- @Override
- public void surfaceDestroyed(SurfaceHolder holder) {
- Log.i(TAG, "SurfaceHolder 被銷毀");
- // 銷毀SurfaceHolder的時(shí)候記錄當(dāng)前的播放位置并停止播放
- if (mediaPlayer != null && mediaPlayer.isPlaying()) {
- currentPosition = mediaPlayer.getCurrentPosition();
- mediaPlayer.stop();
- }
- }
- @Override
- public void surfaceCreated(SurfaceHolder holder) {
- Log.i(TAG, "SurfaceHolder 被創(chuàng)建");
- if (currentPosition > 0) {
- // 創(chuàng)建SurfaceHolder的時(shí)候,如果存在上次播放的位置,則按照上次播放位置進(jìn)行播放
- play(currentPosition);
- currentPosition = 0;
- }
- }
- @Override
- public void surfaceChanged(SurfaceHolder holder, int format, int width,
- int height) {
- Log.i(TAG, "SurfaceHolder 大小被改變");
- }
- };
- private OnSeekBarChangeListener change = new OnSeekBarChangeListener() {
- @Override
- public void onStopTrackingTouch(SeekBar seekBar) {
- // 當(dāng)進(jìn)度條停止修改的時(shí)候觸發(fā)
- // 取得當(dāng)前進(jìn)度條的刻度
- int progress = seekBar.getProgress();
- if (mediaPlayer != null && mediaPlayer.isPlaying()) {
- // 設(shè)置當(dāng)前播放的位置
- mediaPlayer.seekTo(progress);
- }
- }
- @Override
- public void onStartTrackingTouch(SeekBar seekBar) {
- }
- @Override
- public void onProgressChanged(SeekBar seekBar, int progress,
- boolean fromUser) {
- }
- };
- private View.OnClickListener click = new View.OnClickListener() {
- @Override
- public void onClick(View v) {
- switch (v.getId()) {
- case R.id.btn_play:
- play(0);
- break;
- case R.id.btn_pause:
- pause();
- break;
- case R.id.btn_replay:
- replay();
- break;
- case R.id.btn_stop:
- stop();
- break;
- default:
- break;
- }
- }
- };
- /*
- * 停止播放
- */
- protected void stop() {
- if (mediaPlayer != null && mediaPlayer.isPlaying()) {
- mediaPlayer.stop();
- mediaPlayer.release();
- mediaPlayer = null;
- btn_play.setEnabled(true);
- isPlaying = false;
- }
- }
- /**
- * 開始播放
- *
- * @param msec 播放初始位置
- */
- protected void play(final int msec) {
- // 獲取視頻文件地址
- String path = et_path.getText().toString().trim();
- File file = new File(path);
- if (!file.exists()) {
- Toast.makeText(this, "視頻文件路徑錯(cuò)誤", 0).show();
- return;
- }
- try {
- mediaPlayer = new MediaPlayer();
- mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
- // 設(shè)置播放的視頻源
- mediaPlayer.setDataSource(file.getAbsolutePath());
- // 設(shè)置顯示視頻的SurfaceHolder
- mediaPlayer.setDisplay(sv.getHolder());
- Log.i(TAG, "開始裝載");
- mediaPlayer.prepareAsync();
- mediaPlayer.setOnPreparedListener(new OnPreparedListener() {
- @Override
- public void onPrepared(MediaPlayer mp) {
- Log.i(TAG, "裝載完成");
- mediaPlayer.start();
- // 按照初始位置播放
- mediaPlayer.seekTo(msec);
- // 設(shè)置進(jìn)度條的最大進(jìn)度為視頻流的最大播放時(shí)長(zhǎng)
- seekBar.setMax(mediaPlayer.getDuration());
- // 開始線程,更新進(jìn)度條的刻度
- new Thread() {
- @Override
- public void run() {
- try {
- isPlaying = true;
- while (isPlaying) {
- int current = mediaPlayer
- .getCurrentPosition();
- seekBar.setProgress(current);
- sleep(500);
- }
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- }.start();
- btn_play.setEnabled(false);
- }
- });
- mediaPlayer.setOnCompletionListener(new OnCompletionListener() {
- @Override
- public void onCompletion(MediaPlayer mp) {
- // 在播放完畢被回調(diào)
- btn_play.setEnabled(true);
- }
- });
- mediaPlayer.setOnErrorListener(new OnErrorListener() {
- @Override
- public boolean onError(MediaPlayer mp, int what, int extra) {
- // 發(fā)生錯(cuò)誤重新播放
- play(0);
- isPlaying = false;
- return false;
- }
- });
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
- /**
- * 重新開始播放
- */
- protected void replay() {
- if (mediaPlayer != null && mediaPlayer.isPlaying()) {
- mediaPlayer.seekTo(0);
- Toast.makeText(this, "重新播放", 0).show();
- btn_pause.setText("暫停");
- return;
- }
- isPlaying = false;
- play(0);
- }
- /**
- * 暫?;蚶^續(xù)
- */
- protected void pause() {
- if (btn_pause.getText().toString().trim().equals("繼續(xù)")) {
- btn_pause.setText("暫停");
- mediaPlayer.start();
- Toast.makeText(this, "繼續(xù)播放", 0).show();
- return;
- }
- if (mediaPlayer != null && mediaPlayer.isPlaying()) {
- mediaPlayer.pause();
- btn_pause.setText("繼續(xù)");
- Toast.makeText(this, "暫停播放", 0).show();
- }
- }
- }
源碼下載地址 :http://pan.baidu.com/s/1lgKLS