天天看點

Android--視訊播放器

版權聲明:本文為部落客原創文章,轉載請标明出處。 https://blog.csdn.net/chaoyu168/article/details/51179636

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的注釋比較完整.

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添加回調 
sv.getHolder().addCallback(callback); 
// 4.0版本之下需要設定的屬性 
// 設定Surface不維護自己的緩沖區,而是等待螢幕的渲染引擎将内容推送到界面 
// sv.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS); 
// 為進度條添加進度更改事件 
seekBar.setOnSeekBarChangeListener(change); 
} 
  
private Callback callback = new Callback() { 
// SurfaceHolder被修改的時候回調 
@Override 
public void surfaceDestroyed(SurfaceHolder holder) { 
Log.i(TAG, "SurfaceHolder 被銷毀"); 
// 銷毀SurfaceHolder的時候記錄目前的播放位置并停止播放 
if (mediaPlayer != null && mediaPlayer.isPlaying()) { 
currentPosition = mediaPlayer.getCurrentPosition(); 
mediaPlayer.stop(); 
} 
} 
  
@Override 
public void surfaceCreated(SurfaceHolder holder) { 
Log.i(TAG, "SurfaceHolder 被建立"); 
if (currentPosition > 0) { 
// 建立SurfaceHolder的時候,如果存在上次播放的位置,則按照上次播放位置進行播放 
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) { 
// 當進度條停止修改的時候觸發 
// 取得目前進度條的刻度 
int progress = seekBar.getProgress(); 
if (mediaPlayer != null && mediaPlayer.isPlaying()) { 
// 設定目前播放的位置 
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, "視訊檔案路徑錯誤", 0).show(); 
return; 
} 
try { 
mediaPlayer = new MediaPlayer(); 
mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC); 
// 設定播放的視訊源 
mediaPlayer.setDataSource(file.getAbsolutePath()); 
// 設定顯示視訊的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); 
// 設定進度條的最大進度為視訊流的最大播放時長 
seekBar.setMax(mediaPlayer.getDuration()); 
// 開始線程,更新進度條的刻度 
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) { 
// 在播放完畢被回調 
btn_play.setEnabled(true); 
} 
}); 
  
mediaPlayer.setOnErrorListener(new OnErrorListener() { 
  
@Override 
public boolean onError(MediaPlayer mp, int what, int extra) { 
// 發生錯誤重新播放 
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); 
  
} 
  
/** 
* 暫停或繼續 
*/ 
protected void pause() { 
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(); 
} 
  
} 
  
}