Android SurfaceView播放视频源码

移动开发 Android
先来介绍一下大部分软件如何解析一段视频流。首先它需要先确定视频的格式,这个和解码相关, 不同的格式视频编码不同,不是这里的重点。知道了视频的编码格式后,再通过编码格式进行解码,最后得到一帧一帧的图像,并把这些图像快速的显示在界面上, 即为播放一段视频。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. * 
  143. * @param msec 播放初始位置 
  144. */ 
  145. protected void play(final int msec) { 
  146. // 获取视频文件地址 
  147. String path = et_path.getText().toString().trim(); 
  148. File file = new File(path); 
  149. if (!file.exists()) { 
  150. Toast.makeText(this"视频文件路径错误"0).show(); 
  151. return
  152. try { 
  153. mediaPlayer = new MediaPlayer(); 
  154. mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC); 
  155. // 设置播放的视频源 
  156. mediaPlayer.setDataSource(file.getAbsolutePath()); 
  157. // 设置显示视频的SurfaceHolder 
  158. mediaPlayer.setDisplay(sv.getHolder()); 
  159. Log.i(TAG, "开始装载"); 
  160. mediaPlayer.prepareAsync(); 
  161. mediaPlayer.setOnPreparedListener(new OnPreparedListener() { 
  162.   
  163. @Override 
  164. public void onPrepared(MediaPlayer mp) { 
  165. Log.i(TAG, "装载完成"); 
  166. mediaPlayer.start(); 
  167. // 按照初始位置播放 
  168. mediaPlayer.seekTo(msec); 
  169. // 设置进度条的最大进度为视频流的最大播放时长 
  170. seekBar.setMax(mediaPlayer.getDuration()); 
  171. // 开始线程,更新进度条的刻度 
  172. new Thread() { 
  173.   
  174. @Override 
  175. public void run() { 
  176. try { 
  177. isPlaying = true
  178. while (isPlaying) { 
  179. int current = mediaPlayer 
  180. .getCurrentPosition(); 
  181. seekBar.setProgress(current); 
  182. sleep(500); 
  183. catch (Exception e) { 
  184. e.printStackTrace(); 
  185. }.start(); 
  186.   
  187. btn_play.setEnabled(false); 
  188. }); 
  189. mediaPlayer.setOnCompletionListener(new OnCompletionListener() { 
  190.   
  191. @Override 
  192. public void onCompletion(MediaPlayer mp) { 
  193. // 在播放完毕被回调 
  194. btn_play.setEnabled(true); 
  195. }); 
  196.   
  197. mediaPlayer.setOnErrorListener(new OnErrorListener() { 
  198.   
  199. @Override 
  200. public boolean onError(MediaPlayer mp, int what, int extra) { 
  201. // 发生错误重新播放 
  202. play(0); 
  203. isPlaying = false
  204. return false
  205. }); 
  206. catch (Exception e) { 
  207. e.printStackTrace(); 
  208.   
  209.   
  210. /** 
  211. * 重新开始播放 
  212. */ 
  213. protected void replay() { 
  214. if (mediaPlayer != null && mediaPlayer.isPlaying()) { 
  215. mediaPlayer.seekTo(0); 
  216. Toast.makeText(this"重新播放"0).show(); 
  217. btn_pause.setText("暂停"); 
  218. return
  219. isPlaying = false
  220. play(0); 
  221.   
  222.   
  223. /** 
  224. * 暂停或继续 
  225. */ 
  226. protected void pause() { 
  227. if (btn_pause.getText().toString().trim().equals("继续")) { 
  228. btn_pause.setText("暂停"); 
  229. mediaPlayer.start(); 
  230. Toast.makeText(this"继续播放"0).show(); 
  231. return
  232. if (mediaPlayer != null && mediaPlayer.isPlaying()) { 
  233. mediaPlayer.pause(); 
  234. btn_pause.setText("继续"); 
  235. Toast.makeText(this"暂停播放"0).show(); 
  236.   
  237.   

源码下载地址 :http://pan.baidu.com/s/1lgKLS

原文地址: Android,SurfaceView播放视频源码

责任编辑:闫佳明 来源: itbbu
相关推荐

2013-07-01 10:53:05

2015-09-01 16:48:44

ios暴风视频播放器

2015-01-22 15:44:55

Android源码音乐播放器

2022-06-20 05:59:35

5G技术音视频技术安卓系统

2021-08-25 07:43:17

AndroidSurfaceViewTextureView

2011-08-10 15:58:58

iPhone视频

2022-08-16 17:37:06

视频播放器鸿蒙

2013-05-20 17:48:20

2024-03-14 08:24:25

MediaCodec解码播放Android

2013-05-20 17:51:47

Android游戏开发SurfaceView

2013-05-20 17:04:09

2013-08-13 14:08:23

android 4.0WebView

2019-05-09 13:52:19

小米Miui

2011-08-17 14:57:31

iPhone应用视频播放

2011-07-20 16:21:20

iPhone 视频 播放器

2010-06-03 14:53:58

SmokescreenHTML 5Flash

2015-05-21 15:25:42

VLC播放器

2021-01-12 10:58:04

Web视频播

2015-01-19 13:52:38

Android源码多功能播放器

2017-05-10 07:33:41

AndroidWebView视频
点赞
收藏

51CTO技术栈公众号