Android SurfaceView播放视频

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

IBM POWER8通过一年充分融入了“开放”基因,现在有哪些成效?快来注册观看直播,一起拥抱开源的力量。

-->

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

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

时间: 2024-10-12 23:20:03

Android SurfaceView播放视频的相关文章

Android SurfaceView播放视频时横竖屏的调整

对于横屏录制的视频就横屏播放,对于竖屏录制的视频就竖屏播放. 在mainifest文件里对负责播放的Activity添加以下属性“ android:configChanges="orientation|keyboardHidden|screenSize" 重写Acitivity的onConfigurationChanged方法: @Override public void onConfigurationChanged(Configuration newConfig) { super.o

Android开发之MediaPlayer和SurfaceView播放视频

使用MediaPlayer出了了可以播放音频之外,还可以播放视频文件,只不过使用MediaPlayer播放视频时,没有提供图像输出界面.可以使用SurfaceView组件来显示视频图像.使用MediaPlayer和SurfaceView播放视频大致可以分为如下四个步骤: 1)定义SurfaceView组件.最好在布局文件中实现. 2)创建MediaPlayer对象,并为其加载要播放的视频. 3)将所播放的视频画面输出到SurfaceView中.使用MediaPlayer对象的setDisplay

Android实现播放视频

转载:http://www.bdqn.cn/news/201311/12100.shtml 使用VideoView播放视频 VideoView,用于播放一段视频媒体,它继承了SurfaceView,位于"android.widget.VideoView",是一个视频控件. 既然是播放一段视频,那么不可避免的要涉及到一些开始.暂停.停止等操作,VideoView也为开发人员提供了对应的方法,这里简单介绍一些常用的: int getCurrentPosition():获取当前播放的位置.

android 98 MediaPlayer+SurfaceView播放视频

package com.itheima.videoplayer; import java.io.IOException; import android.media.MediaPlayer; import android.os.Bundle; import android.app.Activity; import android.view.Menu; import android.view.SurfaceHolder; import android.view.SurfaceHolder.Callb

请教:Android正播放视频时的解码输出流如何获取?

============问题描述============ Android播放视频,经过解码器解码 获得数据流 再显示到屏幕上.请问这部分数据(解码器解码后的数据流)通过什么方法可以获取?  请教...  先谢谢啦 ============解决方案1============ 你是想录制视频? ============解决方案2============ 引用 3 楼 yu8fei 的回复: Quote: 引用 2 楼 sagittarius1988 的回复: 你是想录制视频? 不是,跟照相机录像机摄

Android OpenGL 播放视频学习

1, 初步接触Open GL: http://www.cnblogs.com/TerryBlog/archive/2010/07/09/1774475.html 使用GLSurfaceView和Render实现一个简单的三角形和正方形.其中,GLSurfaceView用于显示视图,Render用于3D渲染.这个博客的代码,运行时会报: java.lang.IllegalArgumentException: Must use a native order direct Buffer 是因为顶点Bu

使用MediaPlayer和SurfaceView播放视频

使用VideoView播放视频简单.方便,丹有些早期的开发者更喜欢使用MediaPlayer来播放视频,但由于MediaPlayer主要用于播放音频,因此它没有提供图像输出界面,此时 需要借助于SurfaceView来显示MediaPlayer播放的图像输出. 使用MediaPlayer播放视频的步骤如下: 1.创建MediaPlayer对象,并让它加载指定的视频文件. 2.在界面布局文件中定义SurfaceView组件,或在程序中创建SurfaceView组件,并为SurfaceView的Su

Android WebView播放视频(包括全屏播放)

最近项目开发中用到了WebView播放视频的功能,总结了开发中犯过的错误,这些错误在开发是及容易遇到的,所以我这里总结了一下,希望大家看到后不要再犯类似的错误,尽可能提高开发效率: 这个Demo我这里也参考了网上写的一个比较好的一个Demo,经过总结修改,写出来的. 以下是相应代码: MainActivity: package com.androidwebviewdemo; import android.app.Activity; import android.app.ProgressDialo

【转】Android WebView 播放视频总结

今天发现 WebView里播放优酷的视频点击播放按钮后没反应,于是看官方文档和搜索解决,下面是我在别人基础上做的补充: android webView 无法播放视频,无法暂停,继续播放视频问题,无法根据浏览器居中显示内容问题 转自:http://blog.csdn.net/it_ladeng/article/details/8136534 此次遇到一个问题就是webView无法播放视频,查了下谷歌发现可以设置 setting.setPluginsEnabled(true); (从API 11 支