在上一篇博客,Android-MediaPlayer-音频播放-普通准备,介绍了普通准备的播放

一般在开发中,要使用异步准备比较好,因为准备是要去准备硬件来播放,是耗性能的

异步准备和普通准备的区别

  普通准备:一直是主线程,会发生阻塞

  异步准备:主线程 + 一个子线程,不会发生阻塞


MediaPlayer是Android设计的媒体播放器,不仅仅可以播放音频文件,还可以播放视频文件

播放:Audio(音频,.mp3)相关

播放:Video(视频,.mp4)相关

以下图,是Android官方提供:MediaPlayer时序图:

只要会看这个图:就能实现音频/视频播放,暂停,继续,停止,重播,等等

看图规律:

    1.蓝色椭圆形是状态,例如:Initialized已初始化状态,Prepared准备状态,Started启动状态,Stopped停止状态,End结束状态,等等;

    2.单箭头是方法调用:例如:调用reset方法重置,调用prepare方法准备,调用start方法播放,等等;

    3.双箭头是监听回调:例如:onError回调错误,等等


此MediaPlayer播放使用异步准备

package liudeli.my_media1;

import android.database.Cursor;
import android.media.MediaPlayer;
import android.net.Uri;
import android.os.Bundle;
import android.os.SystemClock;
import android.provider.MediaStore;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
import android.widget.Toast; /**
* 此MediaPlayer播放使用使用异步准备,不是普通准备
*/
public class MediaPlayAsyncAudioActivity extends AppCompatActivity { private TextView tvPlayerPath; // 显示播放的路径
private TextView tvAudioInfo; // 歌曲时长/歌手/专辑
private TextView tvAudioThisDuration; // 当前播放的时长
private TextView tv_play_state; // 播放的状态 /**
* 媒体播放器,可以播放(音频/视频)
* 播放(音频/视频)操作一模一样
*/
private MediaPlayer mediaPlayer; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_play_audio); tvPlayerPath = findViewById(R.id.tv_player_path);
tvAudioInfo = findViewById(R.id.tv_audio_info);
tvAudioThisDuration = findViewById(R.id.tv_audio_this_duration);
tv_play_state = findViewById(R.id.tv_play_state); mediaPlayer = new MediaPlayer(); // 为了测试,这样写,真实开发中,不这样写
new Thread(){
@Override
public void run() {
super.run();
while (true) { runOnUiThread(new Runnable() {
@Override
public void run() {
if (mediaPlayer.isPlaying()) {
tvAudioThisDuration.setText("当前时长:" + postions(mediaPlayer.getCurrentPosition()));
}
}
}); SystemClock.sleep(1000);
}
}
}.start(); /**
* 监听播放完成
*/
mediaPlayer.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
@Override
public void onCompletion(MediaPlayer mp) {
Log.d("mp", "播放完成");
Toast.makeText(MediaPlayAsyncAudioActivity.this, "播放完成", Toast.LENGTH_SHORT).show();
tvAudioThisDuration.setText("当前时长:-");
}
}); /**
* 监听播放错误
*/
mediaPlayer.setOnErrorListener(new MediaPlayer.OnErrorListener() {
@Override
public boolean onError(MediaPlayer mp, int what, int extra) {
tv_play_state.setText("播放异常");
return false;
}
}); /**
* 去获取第一条外置存储的音频文件.mp3 的路径
* 通过Uri uri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI; 获取外置存储音频文件
* getContentResolver.query(uri)
*/
initAudioPlayerPath(); tv_play_state.setText("---");
} /**
* 去获取第六条外置存储的音频文件.mp3 的路径
*/
private void initAudioPlayerPath() {
Uri uri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
// 查询的列
String[] projection = new String[]{MediaStore.Audio.Media.DATA, // 音频路径
MediaStore.Audio.Media.DURATION, // 音频时长
MediaStore.Audio.Media.ARTIST, // 歌手
MediaStore.Audio.Media.ALBUM // 专辑
};
// 让Android系统也会去读取外置存储
Cursor cursor = getContentResolver().query(uri,
projection,
null,
null,
null,
null); /**
* 把游标移到第一行:cursor.moveToFirst()
* 把游标移到第六行:cursor.moveToPosition(6)
*/
if (cursor.moveToPosition(6)) {
// if (cursor.moveToFirst()) {
tvPlayerPath.setText(cursor.getString(cursor.getColumnIndex(MediaStore.Audio.Media.DATA)));
String duration = cursor.getString(cursor.getColumnIndex(MediaStore.Audio.Media.DURATION));
String artist = cursor.getString(cursor.getColumnIndex(MediaStore.Audio.Media.ARTIST));
String album = cursor.getString(cursor.getColumnIndex(MediaStore.Audio.Media.ALBUM));
tvAudioInfo.setText("歌曲时长:" + postions(Integer.parseInt(duration)) + " \n歌手:" + artist + " \n专辑:" + album);
}
} /**
* 转换时长值
*/
private String postions(int postion) {
int musicTime = postion / 1000;
return musicTime / 60 + ":" + musicTime % 60;
} /**
* 开始播放:此次播放使用异步准备, 不是普通准备
* @param view
*/
public void player(View view) {
try {
// 重置
mediaPlayer.reset();
// 设置音频文件路径
mediaPlayer.setDataSource(tvPlayerPath.getText().toString().trim());
// 异步准备并播放
asyncPrepare();
} catch (Exception e) {
e.printStackTrace();
}
} /**
* 异步准备的行为
*/
private void asyncPrepare() {
// 准备:是操作硬件在播放,所以需要准备
mediaPlayer.prepareAsync();
// 监听异步准备,一旦准备完成,就会调用此方法
mediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
@Override
public void onPrepared(MediaPlayer mp) {
// 调用此方法,代表异步准备完成✅
// 开始播放
mediaPlayer.start(); tv_play_state.setText("播放中...");
}
});
} /**
* 暂停播放 继续播放
* @param view
*/
public void pause(View view) {
/**
* 这种方式可以拿到控件
*/
Button pause_continue = (Button) view;
if (mediaPlayer.isPlaying()) {
pause_continue.setText("继续");
// 暂停
mediaPlayer.pause(); tv_play_state.setText("暂停中...");
} else {
pause_continue.setText("暂停");
// 继续播放
mediaPlayer.start(); tv_play_state.setText("播放中...");
}
} /**
* 停止播放
*/
public void stop(View view) {
// 停止播放
mediaPlayer.stop(); tv_play_state.setText("---");
} /**
* 重播
* @param view
*/
public void recorded(View view) {
try {
// 先停止
mediaPlayer.stop(); // 异步准备并播放
asyncPrepare();
} catch (Exception e) {
e.printStackTrace();
} finally {
if (mediaPlayer.isPlaying()) {
tv_play_state.setText("播放中...");
}
}
} /**
* 此Activity销毁后,一定要
* mediaPlayer.release();
* mediaPlayer = null;
* 因为 MediaPlayer 是操作硬件在播放,所以一定要释放资源
*/
@Override
protected void onDestroy() {
super.onDestroy();
mediaPlayer.release();
mediaPlayer = null;
System.gc();
}
}

AndroidManifest.xml 配置 外部存储读取权限:

Android系统也会去读取外置存储,需要读取外部存储的权限

  <!--
getContentResolver().query(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, ...)
Android系统也会去读取外置存储,需要读取外部存储的权限
-->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

布局文件:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MediaPlayAudioActivity"
android:orientation="vertical"> <LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:padding="10dp"> <TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="播放音频.mp3路径:"
/> <TextView
android:id="@+id/tv_player_path"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="/mnt/sdcard/"
/> </LinearLayout> <LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"> <Button
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="播放"
android:onClick="player"
/> <Button
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="暂停"
android:onClick="pause"
/> <Button
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="停止"
android:onClick="stop"
/> <Button
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="重播"
android:onClick="recorded"
/> </LinearLayout> <TextView
android:id="@+id/tv_audio_info"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="10dp"
android:text="test"
/> <TextView
android:id="@+id/tv_audio_this_duration"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="10dp"
android:text="歌曲时长:-"
/> <TextView
android:id="@+id/tv_play_state"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textSize="20sp"
android:padding="30dp"
android:layout_gravity="center_horizontal"
/> </LinearLayout>

异步准备性能比普通准备要好:

Android-MediaPlayer-音频播放-异步准备的更多相关文章

  1. Android MediaPlayer 音频倍速播放,调整播放速度

    本文链接: Android MediaPlayer 倍速播放,调整播放速度 现在市面上的很多音视频App都有倍速播放的功能,例如把播放速度调整为0.5.1.5.2倍等等. 从Android API 2 ...

  2. MediaPlayer 音频播放 示例

    状态机.流程图.生命周期 对播放音频/视频文件和流的控制是通过一个状态机来管理的.下图显示一个MediaPlayer对象被支持的播放控制操作驱动的生命周期和状态. 椭圆代表MediaPlayer对象可 ...

  3. Android——简单音乐播放器

    使用MediaPlayer做的简单音乐播放器,更多内容请到百度经验查看   http://jingyan.baidu.com/article/60ccbceb63452364cab197f1.html ...

  4. Android MediaPlayer 播放音频

    本文链接: Android MediaPlayer 播放音频 主要介绍使用MediaPlayer播放音频的方式.关于MediaPlayer的基础知识,比如状态,可以参考Android MediaPla ...

  5. Android MediaPlayer播放一般音频与SoundPool播放短促的音效

    [1]使用MediaPlayer实现一般的音频播放 MediaPlayer播放通常的音频文件 MediaPlayer mediaPlayer = new MediaPlayer(); if (medi ...

  6. Android音频播放之SoundPool

    SoundPool 一.基本概念 在Android应用程序的开发过程中,经常需要播放多媒体文件,也许最先想到的会是MediaPlayer类了,该类提供了播放.暂停.停止及重复播放等功能性方法(该类位于 ...

  7. Android 学习笔记多媒体技术之 AsyncTask+实现音频播放...

    PS:今天搞了一下如何实现音频播放...结果被坑了,看书上写的代码是挺简单的,但是有个函数就是死活没看懂,这真是受不了...最后才弄明白,原来是一个实现异步任务的一个类...这个类使用java.uti ...

  8. android 音频播放总结 soundlPool,MediaPlay

    soundlPool 用于小音频的播放多个同时播放. 使用步骤: 步骤一: 首先下载音频文件可以将其放入assets文件夹下或者res下的raw文件夹下,区别在于assets下可以再新建文件夹二raw ...

  9. 音频播放 音乐 MediaPlayer

    MediaPlayer对象的生命周期如下: Idle 状态:当使用new()方法创建一个MediaPlayer对象或者调用了其reset()方法时,该MediaPlayer对象处于idle状态.这两种 ...

  10. 【Android】20.1 音频播放

    分类:C#.Android.VS2015: 创建日期:2016-03-11 一.简介 MediaPlayer:适合每次播放一个音频资源或者音频文件的场合. SoundPool:适合同时播放多个音频资源 ...

随机推荐

  1. SQL 知识及用法备忘录

    ---查询当前数据库一共有多少张表 ) from sysobjects where xtype='U' ---查询当前数据库有多少张视图 ) from sysobjects where xtype=' ...

  2. Window 端口占用

    Windows平台 在windows命令行窗口下执行: 1.查看所有的端口占用情况 C:\>netstat -ano 协议    本地地址                     外部地址    ...

  3. python + docker, 实现天气数据 从FTP获取以及持久化(三)-- python获取FTP数据

    前言 经过前面两个小节的介绍,我们已经完成了MySQL数据库的搭建和数据库操作的事宜. 在本小节中,我们需要完成的任务是:使用python从FTP服务其上面获取文本文件. 搭建测试FTP服务器 LZ的 ...

  4. J2EE Filter中修改request内容

    最近在做一个微信相关的网站,很多地方涉及到微信表情的输入,导致内容无法插入到数据库,虽然有用到一个表情过滤的工具类,但是需要过滤的地方比较多,于是想到在过滤器中过滤用户请求的内容. request这个 ...

  5. mysql之SQLYog配置

    SQLyog(MySQL图形化开发工具) 安装: 提供的SQLyog软件为免安装版,可直接使用 使用: 输入用户名.密码,点击连接按钮,进行访问MySQL数据库进行操作

  6. ES6系列_2之新的声明方式

    在ES5中我们在声明时只有一种方法,就是使用var来进行声明,ES6对声明的进行了扩展,现在可以有三种声明方式. (1)var:它是variable的简写,可以理解成变量的意思. (2)let:它在英 ...

  7. jQuery常用属性方法大全 attr(),val()

    @@@@属性篇: 写作本篇文章的意义:jQuery的教程千千万,却没有英文版的API讲的系统.到位,一些话用中文翻译过来味道就变了,所以我将英文版的API的一些常用的方法单独提出来放在这里,并用自己的 ...

  8. react之echarts数据更新

    react之echarts数据更新 在使用setState更新数据时,如果要将图标更新,需要做一些简单的封装,代码如下: import React, { Component } from 'react ...

  9. Oracle安装过程出现问题---------安装Oracle11gR2先决条件检查失败

    一.错误信息当安装到“先决条件检查” 时,提示如下图所示的错误: 二.错误原因一般情况下,由于操作系统未开启默认共享,导致Oracle无法检查环境的可用性. 三.解决方法1.在运行中(或键盘按 Win ...

  10. clientX,offsetX,screenX,pageX 区别!

    先总结下区别: event.clientX.event.clientY 鼠标相对于浏览器窗口可视区域的X,Y坐标(窗口坐标),可视区域不包括工具栏和滚动条.IE事件和标准事件都定义了这2个属性 eve ...