Android SurfaceView播放視頻


先來介紹一下大部分軟件如何解析一段視頻流。首先它需要先確定視頻的格式,這個和解碼相關, 不同的格式視頻編碼不同,不是這里的重點。知道了視頻的編碼格式后,再通過編碼格式進行解碼,最后得到一幀一幀的圖像,並把這些圖像快速的顯示在界面上, 即為播放一段視頻。SurfaceView在Android中就是完成這個功能的。

android

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

實現代碼:

  1. package cn.bgxt.surfaceviewdemo; 
  2.   
  3. import java.io.File; 
  4.   
  5. import android.media.AudioManager; 
  6. import android.media.MediaPlayer; 
  7. import android.media.MediaPlayer.OnCompletionListener; 
  8. import android.media.MediaPlayer.OnErrorListener; 
  9. import android.media.MediaPlayer.OnPreparedListener; 
  10. import android.os.Bundle; 
  11. import android.app.Activity; 
  12. import android.util.Log; 
  13. import android.view.SurfaceHolder; 
  14. import android.view.SurfaceHolder.Callback; 
  15. import android.view.SurfaceView; 
  16. import android.view.View; 
  17. import android.widget.Button; 
  18. import android.widget.EditText; 
  19. import android.widget.SeekBar; 
  20. import android.widget.SeekBar.OnSeekBarChangeListener; 
  21. import android.widget.Toast; 
  22.   
  23. public class MainActivity extends Activity { 
  24. private final String TAG = "main"; 
  25. private EditText et_path; 
  26. private SurfaceView sv; 
  27. private Button btn_play, btn_pause, btn_replay, btn_stop; 
  28. private MediaPlayer mediaPlayer; 
  29. private SeekBar seekBar; 
  30. private int currentPosition = 0; 
  31. private boolean isPlaying; 
  32.   
  33. @Override 
  34. protected void onCreate(Bundle savedInstanceState) { 
  35. super.onCreate(savedInstanceState); 
  36. setContentView(R.layout.activity_main); 
  37.   
  38. seekBar = (SeekBar) findViewById(R.id.seekBar); 
  39. sv = (SurfaceView) findViewById(R.id.sv); 
  40. et_path = (EditText) findViewById(R.id.et_path); 
  41.   
  42. btn_play = (Button) findViewById(R.id.btn_play); 
  43. btn_pause = (Button) findViewById(R.id.btn_pause); 
  44. btn_replay = (Button) findViewById(R.id.btn_replay); 
  45. btn_stop = (Button) findViewById(R.id.btn_stop); 
  46.   
  47. btn_play.setOnClickListener(click); 
  48. btn_pause.setOnClickListener(click); 
  49. btn_replay.setOnClickListener(click); 
  50. btn_stop.setOnClickListener(click); 
  51.   
  52. // 為SurfaceHolder添加回調 
  53. sv.getHolder().addCallback(callback); 
  54. // 4.0版本之下需要設置的屬性 
  55. // 設置Surface不維護自己的緩沖區,而是等待屏幕的渲染引擎將內容推送到界面 
  56. // sv.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS); 
  57. // 為進度條添加進度更改事件 
  58. seekBar.setOnSeekBarChangeListener(change); 
  59.   
  60. private Callback callback = new Callback() { 
  61. // SurfaceHolder被修改的時候回調 
  62. @Override 
  63. public void surfaceDestroyed(SurfaceHolder holder) { 
  64. Log.i(TAG, "SurfaceHolder 被銷毀"); 
  65. // 銷毀SurfaceHolder的時候記錄當前的播放位置並停止播放 
  66. if (mediaPlayer != null && mediaPlayer.isPlaying()) { 
  67. currentPosition = mediaPlayer.getCurrentPosition(); 
  68. mediaPlayer.stop(); 
  69.   
  70. @Override 
  71. public void surfaceCreated(SurfaceHolder holder) { 
  72. Log.i(TAG, "SurfaceHolder 被創建"); 
  73. if (currentPosition > 0) { 
  74. // 創建SurfaceHolder的時候,如果存在上次播放的位置,則按照上次播放位置進行播放 
  75. play(currentPosition); 
  76. currentPosition = 0; 
  77.   
  78. @Override 
  79. public void surfaceChanged(SurfaceHolder holder, int format, int width, 
  80. int height) { 
  81. Log.i(TAG, "SurfaceHolder 大小被改變"); 
  82.   
  83. }; 
  84.   
  85. private OnSeekBarChangeListener change = new OnSeekBarChangeListener() { 
  86.   
  87. @Override 
  88. public void onStopTrackingTouch(SeekBar seekBar) { 
  89. // 當進度條停止修改的時候觸發 
  90. // 取得當前進度條的刻度 
  91. int progress = seekBar.getProgress(); 
  92. if (mediaPlayer != null && mediaPlayer.isPlaying()) { 
  93. // 設置當前播放的位置 
  94. mediaPlayer.seekTo(progress); 
  95.   
  96. @Override 
  97. public void onStartTrackingTouch(SeekBar seekBar) { 
  98.   
  99.   
  100. @Override 
  101. public void onProgressChanged(SeekBar seekBar, int progress, 
  102. boolean fromUser) { 
  103.   
  104. }; 
  105.   
  106. private View.OnClickListener click = new View.OnClickListener() { 
  107.   
  108. @Override 
  109. public void onClick(View v) { 
  110.   
  111. switch (v.getId()) { 
  112. case R.id.btn_play: 
  113. play(0); 
  114. break; 
  115. case R.id.btn_pause: 
  116. pause(); 
  117. break; 
  118. case R.id.btn_replay: 
  119. replay(); 
  120. break; 
  121. case R.id.btn_stop: 
  122. stop(); 
  123. break; 
  124. default: 
  125. break; 
  126. }; 
  127.   
  128.   
  129. /* 
  130. * 停止播放 
  131. */ 
  132. protected void stop() { 
  133. if (mediaPlayer != null && mediaPlayer.isPlaying()) { 
  134. mediaPlayer.stop(); 
  135. mediaPlayer.release(); 
  136. mediaPlayer = null; 
  137. btn_play.setEnabled(true); 
  138. isPlaying = false; 
  139.   
  140. /** 
  141. * 開始播放 
  142. * @param msec 播放初始位置 
  143. */ 
  144. protected void play(final int msec) { 
  145. // 獲取視頻文件地址 
  146. String path = et_path.getText().toString().trim(); 
  147. File file = new File(path); 
  148. if (!file.exists()) { 
  149. Toast.makeText(this, "視頻文件路徑錯誤", 0).show(); 
  150. return; 
  151. try { 
  152. mediaPlayer = new MediaPlayer(); 
  153. mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC); 
  154. // 設置播放的視頻源 
  155. mediaPlayer.setDataSource(file.getAbsolutePath()); 
  156. // 設置顯示視頻的SurfaceHolder 
  157. mediaPlayer.setDisplay(sv.getHolder()); 
  158. Log.i(TAG, "開始裝載"); 
  159. mediaPlayer.prepareAsync(); 
  160. mediaPlayer.setOnPreparedListener(new OnPreparedListener() { 
  161.   
  162. @Override 
  163. public void onPrepared(MediaPlayer mp) { 
  164. Log.i(TAG, "裝載完成"); 
  165. mediaPlayer.start(); 
  166. // 按照初始位置播放 
  167. mediaPlayer.seekTo(msec); 
  168. // 設置進度條的最大進度為視頻流的最大播放時長 
  169. seekBar.setMax(mediaPlayer.getDuration()); 
  170. // 開始線程,更新進度條的刻度 
  171. new Thread() { 
  172.   
  173. @Override 
  174. public void run() { 
  175. try { 
  176. isPlaying = true; 
  177. while (isPlaying) { 
  178. int current = mediaPlayer 
  179. .getCurrentPosition(); 
  180. seekBar.setProgress(current); 
  181. sleep(500); 
  182. catch (Exception e) { 
  183. e.printStackTrace(); 
  184. }.start(); 
  185.   
  186. btn_play.setEnabled(false); 
  187. }); 
  188. mediaPlayer.setOnCompletionListener(new OnCompletionListener() { 
  189.   
  190. @Override 
  191. public void onCompletion(MediaPlayer mp) { 
  192. // 在播放完畢被回調 
  193. btn_play.setEnabled(true); 
  194. }); 
  195.   
  196. mediaPlayer.setOnErrorListener(new OnErrorListener() { 
  197.   
  198. @Override 
  199. public boolean onError(MediaPlayer mp, int what, int extra) { 
  200. // 發生錯誤重新播放 
  201. play(0); 
  202. isPlaying = false; 
  203. return false; 
  204. }); 
  205. catch (Exception e) { 
  206. e.printStackTrace(); 
  207.   
  208.   
  209. /** 
  210. * 重新開始播放 
  211. */ 
  212. protected void replay() { 
  213. if (mediaPlayer != null && mediaPlayer.isPlaying()) { 
  214. mediaPlayer.seekTo(0); 
  215. Toast.makeText(this, "重新播放", 0).show(); 
  216. btn_pause.setText("暫停"); 
  217. return; 
  218. isPlaying = false; 
  219. play(0); 
  220.   
  221.   
  222. /** 
  223. * 暫停或繼續 
  224. */ 
  225. protected void pause() { 
  226. if (btn_pause.getText().toString().trim().equals("繼續")) { 
  227. btn_pause.setText("暫停"); 
  228. mediaPlayer.start(); 
  229. Toast.makeText(this, "繼續播放", 0).show(); 
  230. return; 
  231. if (mediaPlayer != null && mediaPlayer.isPlaying()) { 
  232. mediaPlayer.pause(); 
  233. btn_pause.setText("繼續"); 
  234. Toast.makeText(this, "暫停播放", 0).show(); 
  235.   
  236.   

源碼下載地址 :http://pan.baidu.com/s/1lgKLS


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM