在Android上用Canvas绘制音频波形图

最近在研究VAD算法,但调整参数时无法实时看到效果,于是决定将音频波形实时绘制出来,并且语音部分和噪音部分用不同颜色的线条显示,这样就能立即看到VAD算法对各种噪音类型的鲁棒性

为了分而治之,先研究怎么在Android下实时绘制随机生成波形的功能,后面再加入语音获取,以及根据VAD算法的结果用不同颜色线条显示 等功能。

查了下Canvas类,发现drawLines方法跟drawLine方法都是根据4个浮点数绘制一条直线,前者还需要开辟额外的空间,所以选择了后者,结果绘制速度很慢,在小米2s上绘制2秒 单通道16KHz采样率16bit位宽的音频,耗时达到700ms,无奈尝试前者,耗时下降至20ms(最快1ms)

由此也能推测出,后者开销这么大主要是花在drawLine的函数调用,而不是坐标点的生成。

代码

package com.happen23.games.waveviewer;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.util.AttributeSet;
import android.util.Log;
import android.view.View;
import android.view.ViewTreeObserver;

import java.util.Random;

/**
 * Created by Administrator on 2016/5/14.
 */
public class VadView extends View {
    private static final String TAG = "VadView";
    Paint paint;
    int audioSampleNum = 2 * 16000;
    int widthPixels;
    int heightPixels;
    float points[];
    short audio[];

    public VadView(Context context, AttributeSet attrs) {
        super(context, attrs);
        paint = new Paint();
        paint.setColor(Color.BLUE);
        paint.setStrokeJoin(Paint.Join.ROUND);
        paint.setStrokeCap(Paint.Cap.ROUND);
        paint.setStrokeWidth(1);

        getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                widthPixels = getWidth();
                heightPixels = getHeight();
            }
        });
    }
    protected void drawWave2(Canvas canvas, short audio[]) {
        for (int i = 0; i < audioSampleNum-1; i++){
            points[4*i] = (float)i/audioSampleNum * widthPixels;
            points[4*i+1] = heightPixels/2 + (float)audio[i]/32768 * heightPixels/2;
            points[4*i+2] = (float)(i+1)/audioSampleNum * widthPixels;
            points[4*i+3] = heightPixels/2 + (float)audio[i+1]/32768 * heightPixels/2;
        };
        canvas.drawLines(points, paint); //for循环外,效率高
    }
    protected void drawWave1(Canvas canvas, short audio[]){
        float startX, startY, stopX, stopY;
        for (int i = 0; i < audio.length-1; i++){
            startX = (float)i/audioSampleNum * widthPixels;
            startY = heightPixels/2 + (float)audio[i]/32768 * heightPixels/2;
            stopX = (float)(i+1)/audioSampleNum * widthPixels;
            stopY = heightPixels/2 + (float)audio[i+1]/32768 * heightPixels/2;
            canvas.drawLine(startX, startY, stopX, stopY, paint); //for循环内,效率低
        }
    }
    protected void onDraw(Canvas canvas) {
        if (points == null) {
            points = new float[audioSampleNum * 4];
            audio = new short[audioSampleNum];
        }
        Random r = new Random();
        for (int i = 0; i < audioSampleNum; i++){
            audio[i] = (short)(r.nextInt(65536) - 32768);
        }
        Log.w(TAG, "start drawLine");
        drawWave2(canvas, audio);
        Log.w(TAG, "stop drawLine");
        postDelayed(new Runnable() {
            @Override
            public void run() {
                invalidate();
            }
        }, 200);
    }
}

Android音频波形控件是一种可以显示音频波形图的控件。它通常用于音频播放器或录音应用中,可以显示音频文件或实时音频流的波形图形,使用户能够更直观地了解音频的声波变化。 在实现Android音频波形控件时,可以使用Android的绘图功能,通过获取音频数据并将其转换为波形图的形式进行绘制。一般来说,可以通过以下步骤来实现: 1. 获取音频数据:可以通过Android的MediaRecorder类录制音频或使用MediaPlayer类播放音频文件时获取音频数据。也可以使用AudioRecord类从设备的麦克风获取实时音频数据。 2. 转换为波形数据:获取音频数据后,可以将其转换为波形数据,一般是通过将音频数据进行采样和缩放处理得到一系列的波形点。 3. 绘制波形图:使用Android的绘图功能,可以将波形数据绘制波形图。可以使用Canvas类进行绘制,通过绘制线条或样式化的音频波浪形图等形式展示音频波形。 4. 更新和交互:如果需要实时更新波形图,可以使用定时器或回调函数来获取和绘制最新的音频数据。还可以添加交互功能,如缩放、滚动等,以便用户能够更好地查看和操作音频波形。 总之,Android音频波形控件是一种用于显示音频波形图的控件,通过获取音频数据并将其转换为波形数据来进行绘制。它可以应用于多种音频应用中,提供更直观、美观的音频展示效果,提升用户体验。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值