做了个音乐播放器 就一直想做个加一个音乐频谱的展示界面

觉的这是一个好玩的东西,可以将耳边动听的声音形象化,仿佛眼前可以看到声音一样。

但是我在文档的开发者指南里没有讲任何有关音乐频谱的东西,最后还是在google的源码示例中找到了。

你可以直接去参看源代码更原滋原味  以下只是个人对着源码的重构和理解

所有以下所讲的功能,均需要在2.3以上的sdk中才能实现。

音频频谱的获取

首先音频的频谱相关的类叫做 android.media.audiofx.Visualizer;

需要权限    <uses-permission android:name="android.permission.RECORD_AUDIO"/>

所以要做的第一件事 是初始化一个visualizer出来

//使用音乐的sessionId来实例化这个类

mVisualizer
= new Visualizer(mMediaPlayer.getAudioSessionId());
           
//设置每次捕获频谱的大小,音乐在播放中的时候采集的数据的大小或者说是采集的精度吧,我的理解,而且getCaptureSizeRange()所返回的数组里面就两个值
.文档里说数组[0]是最小值(128),数组[1]是最大值(1024)。

mVisualizer.setCaptureSize(Visualizer.getCaptureSizeRange()[1]);

//接下来就好理解了设置一个监听器来监听不断而来的所采集的数据。一共有4个参数,第一个是监听者,第二个单位是毫赫兹,表示的是采集的频率,第三个是是否采集波形,第四个是是否采集频率

mVisualizer.setDataCaptureListener(new
Visualizer.OnDataCaptureListener() {

//这个回调应该采集的是波形数据

@Override
               
public void onWaveFormDataCapture(Visualizer visualizer, byte[]
waveform,
                       
int samplingRate) {

//waveformView 是一个自定义的view用来按照波形来画图 一会后面再讲
                   
waveformView.updateVisualizer(waveform);
               
}

//这个回调应该采集的是快速傅里叶变换有关的数据,没试过,回头有空了再试试

@Override
               
public void onFftDataCapture(Visualizer visualizer, byte[]
fft,
                       
int samplingRate) {
                   
// TODO Auto-generated method stub

}
           
}, Visualizer.getMaxCaptureRate() / 2, true, false);

以上波形的数据采集就完成了,需要注意的一个点是mVisualizer.setEnabled(true);

这个方法的主要作用是为了控制何时去采集频谱数据,你应该只是愿意采集你所关心的音乐数据,而不关心声音输出器中任何的声音。而且对mVisualizer的许多设置必须在setEnable之前完成。并且结束功能后,要记得setEnable(false)

如果你见到了以下这个错误,那基本上就是因为没有及时setEnable(false),导致setCaptureSize()这个方法出错。

E/AndroidRuntime(22259): Caused by:
java.lang.IllegalStateException: setCaptureSize() called in wrong
state: 2

顺带再说一个bug 如果你得到的错误代码是 -1 那么基本上的原因是你忘记了声明权限

<uses-permission
android:name="android.permission.RECORD_AUDIO"/>

音频频谱的展示

你在上一节已经通过监听器获得了波形数据,那么如何展示?这仅仅是一个自定义view的问题,简单废话一下:重点提一下view中的onDraw()方法

@Override
    protected
void onDraw(Canvas canvas) {
       
super.onDraw(canvas);

//mBytes就是采集来的数据 这里是个大小为1024的数组,里面的数据都是byts类型,所以大小为-127到128

if (mBytes == null) {
           
return;
       
}

if (mPoints == null || mPoints.length <
mBytes.length * 4) {

//mPoints主要用来存储要画直线的4个坐标(每个点两个坐标,所以一条直线需要两个点,也就是4个坐标)
           
mPoints = new float[mBytes.length * 4];
       
}

mRect.set(0, 0, getWidth(), getHeight());
//xOrdinate是x轴的总刻度,因为一次会传输过来1024个数据,每两个数据要画成一条直线,所以x轴我们分成1023段。你要是觉的太多了,也可以像我一样除以2,看自己需求了。

int xOrdinate = (mBytes.length - 1)/2;

//以下的for循环将利用mBytes[i] mBytes[i+1]
这两个数据去生成4个坐标值,从而在刻画成两个坐标,来画线条
       
for (int i = 0; i <xOrdinate ; i++) {

//第i个点在总横轴上的坐标,

mPoints[i * 4] = mRect.width() * i / xOrdinate;

//第i个点的在总纵轴上的坐标。他在画线上以总纵轴的1/2为基准线(mRect.height() /
2),所有的点或正或负以此线为基础标记。

//((byte) (mBytes[i] +
128))这个一直没有理解,如果+128是为了将数据全部换算为正整数,那么强转为byte后不又变回-127到128了么??要是谁知道原因可以留言告诉我.....

//(mRect.height() / 2) /
128就是将二分之一的总长度换算成128个刻度,因为我们的数据是byte类型,所以刻画成128个刻度正好
           
mPoints[i * 4 + 1] = mRect.height() / 2+ ((byte) (mBytes[i] + 128))
* (mRect.height() / 2) / 128;

//以下就是刻画第i+1个数据了,原理和刻画第i个一样
           
mPoints[i * 4 + 2] = mRect.width() * (i + 1) / xOrdinate;
           
mPoints[i * 4 + 3] = mRect.height() / 2 + ((byte) (mBytes[i + 1] +
128)) * (mRect.height() / 2) / 128;
       
}

//循环结束后,就得到了这一次波形的所有刻画坐标,直接画在画布上就好了

canvas.drawLines(mPoints, mForePaint);
    }

做的音乐软件 纯粹好玩  就放在了国内的market上

http://market.nduoa.com/apk/detail/412136

android获取和展示音乐的频谱的更多相关文章

  1. Android获取网络状态

    Android获取网络状态 学习自 https://developer.android.google.cn/reference/android/net/ConnectivityManager http ...

  2. Android获取SD卡路径/内存的几种方法

    Android获取SD卡路径 本篇将会带领大家学习如何获取android路径的几种常见用法,但在我开始bb之前需要大家清楚android中内存和外存之间的区别,下面进行简短介绍:android中的内存 ...

  3. Android获取内置sdcard跟外置sdcard路径

    Android获取内置sdcard跟外置sdcard路径.(测试过两个手机,亲测可用) 1.先得到外置sdcard路径,这个接口是系统提供的标准接口. 2.得到上一级文件夹目录 3.得到该目录的所有文 ...

  4. Android获取系统时间方法的总结

    Android获取系统时间方法的方法有很多种,常用的有Calendar.Date.currentTimeMills等方法. (1)Calendar Calendar获取系统时间首先要用Calendar ...

  5. Android获取屏幕宽和高

    android获取屏幕的高度和宽度用到WindowManager这个类,两种方法:   1.WindowManager wm = (WindowManager) getContext()        ...

  6. android 获取文件夹、文件的大小 以B、KB、MB、GB 为单位

    android 获取文件夹.文件的大小 以B.KB.MB.GB 为单位   public class FileSizeUtil { public static final int SIZETYPE_B ...

  7. Android获取APK包名的几种方法

    Android获取APK包名的几种方法:1.adb shell pm list package -f | findstr 关键字 #只能获取到包名,主Activity名无法获取到 2.使用aapt-- ...

  8. I.MX6 android 获取framebuffer信息

    /******************************************************************************** * I.MX6 android 获取 ...

  9. android 获取IMEI号

    android 获取 imei号码 核心代码: Imei = ((TelephonyManager) getSystemService(TELEPHONY_SERVICE)).getDeviceId( ...

随机推荐

  1. C 字符/字符串经常使用函数

    string.h中经常使用函数 char * strchr(char * str ,char ch); 从字符串str中查找首次出现字符ch的位置,若存在返回查找后的地址.若不存在则返回NULL vo ...

  2. 背景剪除和OpenCV中的实现

    转载请注明出处! ! ! http://blog.csdn.net/zhonghuan1992 背景剪除和OpenCV中的实现 背景与前景都是相对的概念.以快速公路为例:有时我们对快速公路上来来往往的 ...

  3. Java – Reading a Large File Efficiently--转

    原文地址:http://www.baeldung.com/java-read-lines-large-file 1. Overview This tutorial will show how to r ...

  4. vue-cli打包项目后,可以修改配置文件

    问题: 前端需要修改后台服务器地址url,写好的配置文件会在npm run build 后压缩在一起,传到运行的前端服务器上后,需要到前端打包的源码,找到url地址进行修改.如果不在打包的源码修改,则 ...

  5. import 与export详解

    ES6 1.export default 其他模块加载该模块时,import命令可以为该匿名函数指定任意名字. 如: import Vue from 'vue' vue里面的第三方模块都是用了这个 使 ...

  6. [RxJS] Avoid mulit post requests by using shareReplay()

    With the shareReplay operator in place, we would no longer fall into the situation where we have acc ...

  7. 跟我一起学extjs5(42--单个模块的数据新增方式)

    跟我一起学extjs5(42--单个模块的数据新增方式)         前面的章节中已经增加了一个自己定义的模块,而且能够进行数据的新增.改动.删除的操作了,在这个基础上就能够大作文章了. 这一节来 ...

  8. Dcloud课程2 什么是Dcloud

    Dcloud课程2  什么是Dcloud 一.总结 一句话总结:DCloud提供了一套快速开发应用的跨平台技术方案. 1.DCloud的产品架构? MUI+(H5+)+HBuilder 2.什么是MU ...

  9. 画pcb时丝印不能再焊盘上

    上图中U3就在焊盘上,这样印出来U3显示不全

  10. [D3] SVG Graphics Containers and Text Elements in D3 v4

    SVG is a great output format for data visualizations because of its scalability, but it comes with s ...