当前位置:  编程技术>移动开发

Android提高之MediaPlayer播放网络视频的实现方法

    来源: 互联网  发布时间:2014-10-26

前面讲解了MediaPlayer播放网络音频,主要介绍了MediaPlayer关于网络音频的缓冲和进度条控制的方法,本文再来讲解一下MediaPlayer播放网络视频的方法。播放网络视频比播放网络音频需要多一个SurfaceView而已,在已经熟悉了MediaPlayer播放网络音频之后,相信大家对本文所述的播放网络视频也能很快地掌握。

先来看看本文程序运行截图,如下所示:

本文程序的视频来自http://daily3gp.com,大家可以替换程序中的视频链接,试试其他影片。

main.xml的源码如下:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
 android:layout_height="fill_parent" android:layout_width="fill_parent">
 <SurfaceView android:id="@+id/surfaceView1"
 android:layout_height="fill_parent" android:layout_width="fill_parent"></SurfaceView>
 <LinearLayout android:layout_height="wrap_content"
 android:layout_width="fill_parent" android:layout_gravity="bottom"
 android:orientation="vertical">
 <LinearLayout android:orientation="horizontal"
  android:layout_gravity="center_horizontal" android:layout_marginTop="4.0dip"
  android:layout_height="wrap_content" android:layout_width="wrap_content">
  <Button android:layout_width="wrap_content"
  android:layout_height="wrap_content" android:id="@+id/btnPlayUrl"
  android:text="播放网络视频"></Button>
  <Button android:layout_height="wrap_content" android:id="@+id/btnPause"
  android:text="暂停" android:layout_width="80dip"></Button>
  <Button android:layout_height="wrap_content"
  android:layout_width="80dip" android:text="停止" android:id="@+id/btnStop"></Button>
 </LinearLayout>
 <LinearLayout android:orientation="horizontal"
  android:layout_width="fill_parent" android:layout_height="wrap_content"
  android:layout_marginBottom="20dip">
  <SeekBar android:paddingRight="10dip" android:layout_gravity="center_vertical"
  android:paddingLeft="10dip" android:layout_weight="1.0"
  android:layout_height="wrap_content" android:layout_width="wrap_content"
  android:id="@+id/skbProgress" android:max="100"></SeekBar>
 </LinearLayout>
 </LinearLayout>
</FrameLayout>

Player.java是本文的核心,Player.java实现了“进度条更新”、“数据缓冲”、“SurfaceHolder生命周期”等功能,其中“SurfaceHolder生命周期”是视频与音频播放的最大区别,通过surfaceCreated()、surfaceDestroyed()、surfaceChanged()可以创建/释放某些资源。下面这个地方需要注意一下:

videoWidth = mediaPlayer.getVideoWidth();
videoHeight = mediaPlayer.getVideoHeight();
if (videoHeight != 0 && videoWidth != 0) {
 arg0.start();
}

有些视频是android播放器不能播放的,不能播放时videoHeight=0,videoWidth=0,以此来判断是否播放视频。

Player.java源码如下:

package com.videoplayer;
import java.io.IOException;
import java.util.Timer;
import java.util.TimerTask;
import android.media.AudioManager;
import android.media.MediaPlayer;
import android.media.MediaPlayer.OnBufferingUpdateListener;
import android.media.MediaPlayer.OnCompletionListener;
import android.os.Handler;
import android.os.Message;
import android.util.Log;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.widget.SeekBar;
public class Player implements OnBufferingUpdateListener,
 OnCompletionListener, MediaPlayer.OnPreparedListener,
 SurfaceHolder.Callback {
 private int videoWidth;
 private int videoHeight;
 public MediaPlayer mediaPlayer;
 private SurfaceHolder surfaceHolder;
 private SeekBar skbProgress;
 private Timer mTimer=new Timer();
 public Player(SurfaceView surfaceView,SeekBar skbProgress)
 {
 this.skbProgress=skbProgress;
 surfaceHolder=surfaceView.getHolder();
 surfaceHolder.addCallback(this);
 surfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
 mTimer.schedule(mTimerTask, 0, 1000);
 }
 /*******************************************************
 * 通过定时器和Handler来更新进度条
 ******************************************************/
 TimerTask mTimerTask = new TimerTask() {
 @Override
 public void run() {
  if(mediaPlayer==null)
  return;
  if (mediaPlayer.isPlaying() && skbProgress.isPressed() == false) {
  handleProgress.sendEmptyMessage(0);
  }
 }
 };
 Handler handleProgress = new Handler() {
 public void handleMessage(Message msg) {
  int position = mediaPlayer.getCurrentPosition();
  int duration = mediaPlayer.getDuration();
  
  if (duration > 0) {
  long pos = skbProgress.getMax() * position / duration;
  skbProgress.setProgress((int) pos);
  }
 };
 };
 //*****************************************************
 public void play()
 {
 mediaPlayer.start();
 }
 public void playUrl(String videoUrl)
 {
 try {
  mediaPlayer.reset();
  mediaPlayer.setDataSource(videoUrl);
  mediaPlayer.prepare();//prepare之后自动播放
  //mediaPlayer.start();
 } catch (IllegalArgumentException e) {
  // TODO Auto-generated catch block
  e.printStackTrace();
 } catch (IllegalStateException e) {
  // TODO Auto-generated catch block
  e.printStackTrace();
 } catch (IOException e) {
  // TODO Auto-generated catch block
  e.printStackTrace();
 }
 }
 public void pause()
 {
 mediaPlayer.pause();
 }
 public void stop()
 {
 if (mediaPlayer != null) { 
  mediaPlayer.stop();
      mediaPlayer.release(); 
      mediaPlayer = null; 
    } 
 }
 @Override
 public void surfaceChanged(SurfaceHolder arg0, int arg1, int arg2, int arg3) {
 Log.e("mediaPlayer", "surface changed");
 }
 @Override
 public void surfaceCreated(SurfaceHolder arg0) {
 try {
  mediaPlayer = new MediaPlayer();
  mediaPlayer.setDisplay(surfaceHolder);
  mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
  mediaPlayer.setOnBufferingUpdateListener(this);
  mediaPlayer.setOnPreparedListener(this);
 } catch (Exception e) {
  Log.e("mediaPlayer", "error", e);
 }
 Log.e("mediaPlayer", "surface created");
 }
 @Override
 public void surfaceDestroyed(SurfaceHolder arg0) {
 Log.e("mediaPlayer", "surface destroyed");
 }
 @Override
 /**
 * 通过onPrepared播放
 */
 public void onPrepared(MediaPlayer arg0) {
 videoWidth = mediaPlayer.getVideoWidth();
 videoHeight = mediaPlayer.getVideoHeight();
 if (videoHeight != 0 && videoWidth != 0) {
  arg0.start();
 }
 Log.e("mediaPlayer", "onPrepared");
 }

 @Override
 public void onCompletion(MediaPlayer arg0) {
 // TODO Auto-generated method stub
 }
 @Override
 public void onBufferingUpdate(MediaPlayer arg0, int bufferingProgress) {
 skbProgress.setSecondaryProgress(bufferingProgress);
 int currentProgress=skbProgress.getMax()*mediaPlayer.getCurrentPosition()/mediaPlayer.getDuration();
 Log.e(currentProgress+"% play", bufferingProgress + "% buffer");
 }
}

test_videoplayer.java是主程序,负责调用Player类,其中关键部分是SeekBarChangeEvent这个SeekBar拖动的事件:SeekBar的Progress是0~SeekBar.getMax()之内的数,而MediaPlayer.seekTo()的参数是0~MediaPlayer.getDuration()之内数,所以MediaPlayer.seekTo()的参数是(progress/seekBar.getMax())*MediaPlayer.getDuration()。

test_videoplayer.java源码如下:

package com.videoplayer;
import android.app.Activity;
import android.content.pm.ActivityInfo;
import android.net.Uri;
import android.os.Bundle;
import android.util.Log;
import android.view.SurfaceView;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.SeekBar;
public class test_videoplayer extends Activity {
 private SurfaceView surfaceView;
 private Button btnPause, btnPlayUrl, btnStop;
 private SeekBar skbProgress;
 private Player player;
 /** Called when the activity is first created. */
 @Override
 public void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.main);
 setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
 surfaceView = (SurfaceView) this.findViewById(R.id.surfaceView1);

 btnPlayUrl = (Button) this.findViewById(R.id.btnPlayUrl);
 btnPlayUrl.setOnClickListener(new ClickEvent());

 btnPause = (Button) this.findViewById(R.id.btnPause);
 btnPause.setOnClickListener(new ClickEvent());

 btnStop = (Button) this.findViewById(R.id.btnStop);
 btnStop.setOnClickListener(new ClickEvent());

 skbProgress = (SeekBar) this.findViewById(R.id.skbProgress);
 skbProgress.setOnSeekBarChangeListener(new SeekBarChangeEvent());
 player = new Player(surfaceView, skbProgress);
 }
 class ClickEvent implements OnClickListener {
 @Override
 public void onClick(View arg0) {
  if (arg0 == btnPause) {
  player.pause();
  } else if (arg0 == btnPlayUrl) {
  String url="http://daily3gp.com/vids/family_guy_penis_car.3gp";
  player.playUrl(url);
  } else if (arg0 == btnStop) {
  player.stop();
  }
 }
 }
 class SeekBarChangeEvent implements SeekBar.OnSeekBarChangeListener {
 int progress;
 @Override
 public void onProgressChanged(SeekBar seekBar, int progress,
  boolean fromUser) {
  // 原本是(progress/seekBar.getMax())*player.mediaPlayer.getDuration()
  this.progress = progress * player.mediaPlayer.getDuration()
   / seekBar.getMax();
 }
 @Override
 public void onStartTrackingTouch(SeekBar seekBar) {

 }
 @Override
 public void onStopTrackingTouch(SeekBar seekBar) {
  // seekTo()的参数是相对与影片时间的数字,而不是与seekBar.getMax()相对的数字
  player.mediaPlayer.seekTo(progress);
 }
 }
}

感兴趣的读者可以自己动手测试一下本文实例代码,相信对大家进行Android程序开发有一定的借鉴作用。


    
相关技术文章:
    ▪Android提高之手游转电视游戏的模拟操控

     目前智能电视终端(智能电视和智能电视盒子)已经越来越火,过去主打视频功能,如今的智能电视终端不仅会继续完善视频功能,还会加入电视游戏功能,同时这也赶上了“电视游戏机解禁”的时机。 当今的大部分Android手游都能够在Android系统的电视终端上运行,其中有少数手游是原生支持手柄(例如MOGA手柄),这部分游戏可以作为电视游戏。但其他手游(射击,赛车,动作等游戏)若要在电视上玩,就需要修改操控模式,把触摸屏操控改为手柄实体键操控。 本文主要讲解的是如何使用/system/bin/之下的Input命令模拟按键和触摸屏操作,调用Inp......


    ▪Android提高之Android手机与BLE终端通信

     最近穿戴设备发展得很火,把相关技术也带旺了,其中一项是BLE(Bluetooth Low Energy)。BLE是蓝牙4.0的核心Profile,主打功能是快速搜索,快速连接,超低功耗保持连接和传输数据,弱点是数据传输速率低,由于BLE的低功耗特点,因此普遍用于穿戴设备。Android 4.3才开始支持BLE API,所以请各位客官把本文代码运行在蓝牙4.0和Android 4.3及其以上的系统,另外本文所用的BLE终端是一个蓝牙4.0的串口蓝牙模块。 注:笔者的i9100刷了4.4系统后,竟然也能跟BLE蓝牙模块通信。 BLE分为三部分Service、Characteristic、Descriptor,这三部分都由UUID作为唯一标示符。一个蓝牙4.0的......


    ▪Android提高之BLE开发Android手机搜索iBeacon基站

     前面文章讲述了Android手机与BLE终端之间的通信,而最常见的BLE终端应该是苹果公司倡导的iBeacon基站。iBeacon技术基于BLE,它的特点是通过广播对外发送消息,手机不需要连上iBeacon基站也能获取它的信息,目前主要用来做室内定位和营销信息推送,在BLE发出的广播里带上带上特定的信息从而被识别为iBeacon。在iOS里面使用iBeacon要经过passbook注册iBeacon的UUID和对应的文字简介,而在Android上则无类似passbook这种系统级的后台蓝牙搜索服务,这类服务需要开发者自己搞定。这次就讲讲Android手机搜索基站和显示iBeacon基站参数。 搜索iBeacon基站的不难,核心在于BluetoothAda......


 
最新技术文章:
    ▪Android开发之登录验证实例教程

     本文所述实例源自一个项目开发中的登录验证功能,具体的要求就是,在Android端输入用户名和密码,在服务器端验证MySQL数据库中是否有此用户,实现之前当然首要的是,如何使Android端的数据发送到服务器端,具体的实现方法如下: 服务器端:ManageServlet.java代码如下: public class ManageServlet extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { request.setCharacterEncoding("utf-8"); response.setCharacterEncoding("utf-8"); String name = request.getParameter("name"); String password......


    ▪Android开发之注册登录方法示例

     本文所述,继续上一篇关于Android端向服务器端发送数据的方法进一步完善注册登录的方法,由于版本问题出现一点瑕疵,今天经过调试已经解决,在这里给大家介绍一下。 在Android4.0以后版本的对于网络权限要求变得严格,致使上一篇所述的案例无法将数据发送到服务器端,当你一点击发送数据,Android控制台就会报错,错误当然是很让人头疼,基本上都是关于http的错误,所以可以肯定是Android虚拟机向服务器发送数据时出现了错误,经过一番检查与测试后才知道,4.0之后的版本,主线程中不允许调用网络,如果你需要的话,应该另起一个线程处理网络连接Th......


    ▪Android获取手机SIM卡运营商信息的方法

     本文实例讲述了Android获取手机SIM卡运营商信息的方法,对于Android程序设计有非常实用的价值。分享给大家供大家参考之用。具体方法如下: 主要功能代码如下: /** * 获取SIM卡运营商 * * @param context * @return */ public static String getOperators(Context context) { TelephonyManager tm = (TelephonyManager) context .getSystemService(Context.TELEPHONY_SERVICE); String operator = null; String IMSI = tm.getSubscriberId(); if (IMSI == null || IMSI.equals("")) { return operator; } if (IMSI.startsWith("46000") || IMSI.startsWith("46002")) { operator = "中国移动"; } else if (IMSI.startsWith("4......


 


站内导航:


特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

©2012-2015,169IT.COM,站长邮箱:www_169it_com#163.com(请将#改为@)

浙ICP备11055608号