热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Android多媒体之MediaPlayer

Android的MediaPlayer包含了Audio和video的播放功能,在Android的界面上,Music和Video两个应用程序都是调用Med


Android的MediaPlayer包含了Audio和video的播放功能,在Android的界面上,Music和Video两个应用程序都是调用MediaPlayer实现的。


 


MediaPlayer的生命周期


 



 


这张状态转换图清晰的描述了 MediaPlayer 的各个状态,也列举了主要的方法的调用时序,每种方法只能在一些特定的状态下使用,如果使用时 MediaPlayer 的状态不正确则会引发 IllegalStateException 异常 。


 


Idle 状态: 当使用 new() 方法创建一个 MediaPlayer 对象或者调用了其 reset() 方法时,该 MediaPlayer 对象处于 idle 状态。这两种方法的一个重要差别就是:如果在这个状态下调用了 getDuration() 等方法(相当于调用时机不正确),通过 reset() 方法进入 idle 状态的话会触发 OnErrorListener.onError() ,并且 MediaPlayer 会进入 Error 状态;如果是新创建的 MediaPlayer 对象,则并不会触发 onError(), 也不会进入 Error 状态。


 


End 状态: 通过 release() 方法可以进入 End 状态,只要 MediaPlayer 对象不再被使用,就应当尽快将其通过 release() 方法释放掉,以释放相关的软硬件组件资源,这其中有些资源是只有一份的(相当于临界资源)。如果 MediaPlayer 对象进入了 End 状态,则不会在进入任何其他状态了。


 


Initialized 状态: 这个状态比较简单, MediaPlayer 调用 setDataSource() 方法就进入 Initialized 状态,表示此时要播放的文件已经设置好了。


 


Prepared 状态: 初始化完成之后还需要通过调用 prepare() 或 prepareAsync() 方法,这两个方法一个是同步的一个是异步的,只有进入 Prepared 状态,才表明 MediaPlayer 到目前为止都没有错误,可以进行文件播放。


 


Preparing 状态: 这个状态比较好理解,主要是和 prepareAsync() 配合,如果异步准备完成,会触发 OnPreparedListener.onPrepared() ,进而进入 Prepared 状态。


 


Started 状态: 显然, MediaPlayer 一旦准备好,就可以调用 start() 方法,这样 MediaPlayer 就处于 Started 状态,这表明 MediaPlayer 正在播放文件过程中。可以使用 isPlaying() 测试 MediaPlayer 是否处于了 Started 状态。 如果播放完毕,而又设置了循环播放,则 MediaPlayer 仍然会处于 Started 状态,类似的,如果在该状态下 MediaPlayer 调用了 seekTo() 或者 start() 方法均可以让 MediaPlayer 停留在 Started 状态。


 


Paused 状态: Started 状态下 MediaPlayer 调用 pause() 方法可以暂停 MediaPlayer ,从而进入 Paused 状态, MediaPlayer 暂停后再次调用 start() 则可以继续 MediaPlayer 的播放,转到 Started 状态,暂停状态时可以调用 seekTo() 方法,这是不会改变状态的。


 


Stop 状态: Started 或者 Paused 状态下均可调用 stop() 停止 MediaPlayer ,而处于 Stop 状态的 MediaPlayer 要想重新播放,需要通过 prepareAsync() 和 prepare() 回到先前的 Prepared 状态重新开始才可以。


 


PlaybackCompleted 状态: 文件正常播放完毕,而又没有设置循环播放的话就进入该状态,并会触发 OnCompletionListener 的 onCompletion() 方法。此时可以调用 start() 方法重新从头播放文件,也可以 stop() 停止 MediaPlayer ,或者也可以 seekTo() 来重新定位播放位置。


 


Error 状态: 如果由于某种原因 MediaPlayer 出现了错误,会触发 OnErrorListener.onError() 事件,此时 MediaPlayer 即进入 Error 状态,及时捕捉并妥善处理这些错误是很重要的,可以帮助我们及时释放相关的软硬件资源,也可以改善用户体验。通过 setOnErrorListener(android.media.MediaPlayer.OnErrorListener) 可以设置该监听器。如果 MediaPlayer 进入了 Error 状态,可以通过调用 reset() 来恢复,使得 MediaPlayer 重新返回到 Idle 状态。


 


下面的程序是使用MediaPlayer播放音频文件并通过SeekBar显示进度的程序:


 





package com.example.musicplay;
import java.io.IOException;
import java.util.Timer;
import java.util.TimerTask;
import android.annotation.SuppressLint;
import android.app.Activity;
import android.media.MediaPlayer;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.SeekBar;
//import android.widget.SeekBar.OnSeekBarChangeListener;

@SuppressLint(
"HandlerLeak")
public class MusicActivity extends Activity
{
private MediaPlayer mp;
private Button play;
private Button pause;
private Button speed;
private Button stop;
private SeekBar seekbar;

private Handler myHandler;

@SuppressLint(
"HandlerLeak")
@Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_music);

initMusic();
initPlayerControl();
initHandler();

new Timer().schedule(new TimerTask()
{
@Override
public void run()
{
if(mp == null)
return;
if(mp.isPlaying()&&!seekbar.isPressed())
{
Message msg
= new Message();
msg.what
= 0x123;
myHandler.sendMessage(msg);
}
}
},
0,1000);
}

//初始化Handle
private void initHandler()
{
myHandler
= new Handler()
{
@Override
public void handleMessage(Message msg)
{
if(msg.what == 0x123)
{
int position = mp.getCurrentPosition();
seekbar.setProgress(position );
}
}

};
}
//初始化mp,加载音乐
private void initMusic()
{
mp
= MediaPlayer.create(this, R.raw.xxx);
mp.setLooping(
false);
}
//初始化播放器控件
private void initPlayerControl()
{
play
= (Button)findViewById(R.id.play);
pause
= (Button)findViewById(R.id.pause);
speed
= (Button)findViewById(R.id.speed);
stop
= (Button)findViewById(R.id.stop);
seekbar
= (SeekBar)findViewById(R.id.my_seekbar);

play.setEnabled(
true);
pause.setEnabled(
false);
speed.setEnabled(
false);
stop.setEnabled(
false);
initSeekBar();

play.setOnClickListener(
new OnClickListener()
{
@Override
public void onClick(View v)
{
if(mp!=null)
{
mp.start();
play.setEnabled(
false);
pause.setEnabled(
true);
speed.setEnabled(
true);
stop.setEnabled(
true);
}
}
});

pause.setOnClickListener(
new OnClickListener()
{
@Override
public void onClick(View v)
{
if(mp!=null)
{
mp.pause();
play.setEnabled(
true);
pause.setEnabled(
false);
speed.setEnabled(
false);
stop.setEnabled(
false);
}
}
});

speed.setOnClickListener(
new OnClickListener()
{
@Override
public void onClick(View v)
{
if(mp!=null)
{
int positon = mp.getCurrentPosition();
int duration = mp.getDuration();
int headon = duration/20;
mp.seekTo(positon
+headon);
}
}
});

stop.setOnClickListener(
new OnClickListener()
{
@Override
public void onClick(View v)
{
if(mp!=null)
{
mp.stop();
play.setEnabled(
true);
pause.setEnabled(
false);
speed.setEnabled(
false);
stop.setEnabled(
false);
try
{
mp.prepare();
mp.seekTo(
0);
seekbar.setProgress(
0);
}
catch (IllegalStateException e)
{
e.printStackTrace();
}
catch (IOException e)
{
e.printStackTrace();
}
}
}
});
}

//初始化seekbar拖拽效果
private void initSeekBar()
{
seekbar.setMax(mp.getDuration());
// seekbar.setOnSeekBarChangeListener(new OnSeekBarChangeListener()
// {
// @Override
// public void onStopTrackingTouch(SeekBar seekBar)
// {
// }
//
// @Override
// public void onStartTrackingTouch(SeekBar seekBar)
// {
// }
//
// @Override
// public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser)
// {
// if(mp == null)
// return;
// mp.seekTo(progress);
// }
// });
}
@Override
protected void onDestroy()
{
if(mp !=null)
mp.release();
}


}



<LinearLayout xmlns:android&#61;"http://schemas.android.com/apk/res/android"
xmlns:tools
&#61;"http://schemas.android.com/tools"
android:layout_width
&#61;"match_parent"
android:layout_height
&#61;"match_parent"
android:orientation
&#61;"vertical" >
<LinearLayout
android:layout_width&#61;"match_parent"
android:layout_height
&#61;"wrap_content"
android:orientation
&#61;"horizontal" >
<Button
android:id&#61;"&#64;&#43;id/play"
android:layout_width
&#61;"wrap_content"
android:layout_height
&#61;"wrap_content"
android:layout_weight
&#61;"1"
android:text
&#61;"play" />
<Button
android:id&#61;"&#64;&#43;id/pause"
android:layout_width
&#61;"wrap_content"
android:layout_height
&#61;"wrap_content"
android:layout_weight
&#61;"1"
android:text
&#61;"pause" />
<Button
android:id&#61;"&#64;&#43;id/speed"
android:layout_width
&#61;"wrap_content"
android:layout_height
&#61;"wrap_content"
android:layout_weight
&#61;"1"
android:text
&#61;"speed" />
<Button
android:id&#61;"&#64;&#43;id/stop"
android:layout_width
&#61;"wrap_content"
android:layout_height
&#61;"wrap_content"
android:layout_weight
&#61;"1"
android:text
&#61;"stop" />
LinearLayout>

<SeekBar
android:id&#61;"&#64;&#43;id/my_seekbar"
android:paddingLeft
&#61;"10dp"
android:paddingRight
&#61;"10dp"
android:layout_width
&#61;"match_parent"
android:layout_height
&#61;"wrap_content"
android:max
&#61;"100">
SeekBar>
LinearLayout>


 



 


 


转载于:https://www.cnblogs.com/Bigmouse123/p/3345706.html



推荐阅读
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • Android开发实现的计时器功能示例
    本文分享了Android开发实现的计时器功能示例,包括效果图、布局和按钮的使用。通过使用Chronometer控件,可以实现计时器功能。该示例适用于Android平台,供开发者参考。 ... [详细]
  • 本文介绍了pack布局管理器在Perl/Tk中的使用方法及注意事项。通过调用pack()方法,可以控制部件在显示窗口中的位置和大小。同时,本文还提到了在使用pack布局管理器时,应注意将部件分组以便在水平和垂直方向上进行堆放。此外,还介绍了使用Frame部件或Toplevel部件来组织部件在窗口内的方法。最后,本文强调了在使用pack布局管理器时,应避免在中间切换到grid布局管理器,以免造成混乱。 ... [详细]
  • android 触屏处理流程,android触摸事件处理流程 ? FOOKWOOD「建议收藏」
    android触屏处理流程,android触摸事件处理流程?FOOKWOOD「建议收藏」最近在工作中,经常需要处理触摸事件,但是有时候会出现一些奇怪的bug,比如有时候会检测不到A ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • 本文整理了Java中org.gwtbootstrap3.client.ui.Icon.addDomHandler()方法的一些代码示例,展示了Icon.ad ... [详细]
  • 目录实现效果:实现环境实现方法一:基本思路主要代码JavaScript代码总结方法二主要代码总结方法三基本思路主要代码JavaScriptHTML总结实 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • Android JSON基础,音视频开发进阶指南目录
    Array里面的对象数据是有序的,json字符串最外层是方括号的,方括号:[]解析jsonArray代码try{json字符串最外层是 ... [详细]
  • 今日份分享:Flutter自定义之旋转木马
    今日份分享:Flutter自定义之旋转木马-先上图,带你回到童年时光:效果分析子布局按照圆形顺序放置且平分角度子布局旋转、支持手势滑动旋转、快速滑动抬手继续旋转、自动旋转支持X轴旋 ... [详细]
  • 详解Android  自定义UI模板设计_由浅入深
    学习安卓已有一些日子,前段时间整理了不少笔记,但是发现笔记不变分享与携带。今天开始整理博客,全当是与大家分享交流与自身学习理解的过程吧。结合最近在做的一个新闻类app及学习中的问题,一点一点整理一下, ... [详细]
  • 第一步:PyQt4Designer设计程序界面该部分设计类同VisvalStudio内的设计,改下各部件的objectName!设计 ... [详细]
  • 在一对一直播源码使用过程中,有时会出现软键盘切换闪屏问题,就是当切换表情的时候屏幕会跳动,因此要对一对一直播源码表情面板无缝切换进行优化。 ... [详细]
author-avatar
个性2402852463
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有