热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

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

这篇文章主要介绍了Android的MediaPlayer播放网络音频的实现方法,很实用的功能,需要的朋友可以参考下

前面有文章曾经地介绍过MediaPlayer的基本用法,这里就更加深入地讲解MediaPlayer的在线播放功能。本文主要实现MediaPlayer在线播放音频的功能,由于在线视频播放比在线音频播放复杂,因此先介绍在线音频播放的实现,这样可以帮助大家逐步深入了解MediaPlayer的在线播放功能。

先来看看本文程序运行的结果如下图所示:

main.xml的源码如下:

<&#63;xml version="1.0" encoding="utf-8"&#63;>

Player.java是本文的核心,Player.java实现了“进度条更新”、“数据缓冲”等功能,虽然不是很复杂的功能,但却是非常有用的功能。

Player.java源码如下:

package com.musicplayer;
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.widget.SeekBar;
public class Player implements OnBufferingUpdateListener,
 OnCompletionListener, MediaPlayer.OnPreparedListener{
 public MediaPlayer mediaPlayer;
 private SeekBar skbProgress;
 private Timer mTimer=new Timer();
 public Player(SeekBar skbProgress)
 {
 this.skbProgress=skbProgress;
 try {
  mediaPlayer = new MediaPlayer();
  mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
  mediaPlayer.setOnBufferingUpdateListener(this);
  mediaPlayer.setOnPreparedListener(this);
 } catch (Exception e) {
  Log.e("mediaPlayer", "error", e);
 }
 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
 /**
 * 通过onPrepared播放
 */
 public void onPrepared(MediaPlayer arg0) {
 arg0.start();
 Log.e("mediaPlayer", "onPrepared");
 }
 @Override
 public void onCompletion(MediaPlayer arg0) {
 Log.e("mediaPlayer", "onCompletion");
 }
 @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_musicplayer.java是主程序,负责调用Player类,其中关键部分是SeekBarChangeEvent这个SeekBar拖动的事件:SeekBar的Progress是0~SeekBar.getMax()之内的数,而MediaPlayer.seekTo()的参数是0~MediaPlayer.getDuration()之内数,所以MediaPlayer.seekTo()的参数是(progress/seekBar.getMax())*player.mediaPlayer.getDuration()。

test_musicplayer.java源码如下:

package com.musicplayer;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.SeekBar;
public class test_musicplayer extends Activity {
 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);
 this.setTitle("在线音乐播放---hellogv编写");
 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(skbProgress);
 }
 class ClickEvent implements OnClickListener {
 @Override
 public void onClick(View arg0) {
  if (arg0 == btnPause) {
  player.pause();
  } else if (arg0 == btnPlayUrl) {
  //在百度MP3里随便搜索到的,大家可以试试别的链接
  String url="http://219.138.125.22/myweb/mp3/CMP3/JH19.MP3";
  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 的 Canvas 和 View 组件创建一个简单的绘图板应用程序,支持触摸绘画和保存图片功能。 ... [详细]
  • C#设计模式学习笔记:观察者模式解析
    本文将探讨观察者模式的基本概念、应用场景及其在C#中的实现方法。通过借鉴《Head First Design Patterns》和维基百科等资源,详细介绍该模式的工作原理,并提供具体代码示例。 ... [详细]
  • Appium + Java 自动化测试中处理页面空白区域点击问题
    在进行移动应用自动化测试时,有时会遇到某些页面没有返回按钮,只能通过点击空白区域返回的情况。本文将探讨如何在Appium + Java环境中有效解决此类问题,并提供详细的解决方案。 ... [详细]
  • 如何清除Chrome浏览器地址栏的特定历史记录
    在使用Chrome浏览器时,你可能会发现地址栏保存了大量浏览记录。有时你可能希望删除某些特定的历史记录而不影响其他数据。本文将详细介绍如何单独删除地址栏中的特定记录以及批量清除所有历史记录的方法。 ... [详细]
  • 利用Selenium与ChromeDriver实现豆瓣网页全屏截图
    本文介绍了一种使用Selenium和ChromeDriver结合Python代码,轻松实现对豆瓣网站进行完整页面截图的方法。该方法不仅简单易行,而且解决了新版Selenium不再支持PhantomJS的问题。 ... [详细]
  • 探索新一代API文档工具,告别Swagger的繁琐
    对于后端开发者而言,编写和维护API文档既繁琐又不可或缺。本文将介绍一款全新的API文档工具,帮助团队更高效地协作,简化API文档生成流程。 ... [详细]
  • 本文详细介绍了Grand Central Dispatch (GCD) 的核心概念和使用方法,探讨了任务队列、同步与异步执行以及常见的死锁问题。通过具体示例和代码片段,帮助开发者更好地理解和应用GCD进行多线程开发。 ... [详细]
  • 本文探讨了在构建应用程序时,如何对不同类型的数据进行结构化设计。主要分为三类:全局配置、用户个人设置和用户关系链。每种类型的数据都有其独特的用途和应用场景,合理规划这些数据结构有助于提升用户体验和系统的可维护性。 ... [详细]
  • 本文将详细介绍多个流行的 Android 视频处理开源框架,包括 ijkplayer、FFmpeg、Vitamio、ExoPlayer 等。每个框架都有其独特的优势和应用场景,帮助开发者更高效地进行视频处理和播放。 ... [详细]
  • 深入理解ExtJS:从入门到精通
    本文详细介绍了ExtJS的功能及其在大型企业前端开发中的应用。通过实例和详细的文件结构解析,帮助初学者快速掌握ExtJS的核心概念,并提供实用技巧和最佳实践。 ... [详细]
  • Android 6.0 切换指定 Wi-Fi 的解决方案
    本文详细介绍了在 Android 6.0 系统中切换到指定 Wi-Fi 的方法,包括常见的问题、原因分析及解决方案。通过官方文档和代码示例,帮助开发者更好地理解和实现这一功能。 ... [详细]
  • Python自动化测试入门:Selenium环境搭建
    本文详细介绍如何在Python环境中安装和配置Selenium,包括开发工具PyCharm的安装、Python环境的设置以及Selenium包的安装方法。此外,还提供了编写和运行第一个自动化测试脚本的步骤。 ... [详细]
  • 本文详细介绍如何在 iOS 7 环境下申请苹果开发者账号,涵盖从访问开发者网站到最终激活账号的完整流程。包括选择个人或企业账号类型、付款方式及注意事项等。 ... [详细]
  • 本文介绍了如何通过Java代码计算一个整数的位数,并展示了多个基础编程示例,包括求和、平均分计算、条件判断等。 ... [详细]
  • 本题要求在一组数中反复取出两个数相加,并将结果放回数组中,最终求出最小的总加法代价。这是一个经典的哈夫曼编码问题,利用贪心算法可以有效地解决。 ... [详细]
author-avatar
mobiledu2502924873
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有