libasound+libsndfile读取麦克风音频保存为wav文件

依赖库

要完成该功能依赖libasound、libsndfile库,该命令在linux下可以通过命令行安装安装,安装命令如下:

bash 复制代码
sudo apt-get install libasound2-dev
sudo apt-get install libsndfile1-dev

代码实现

本文的目的是在linux下利用libasound和libsndfile库基于C++实现读取麦克风数据,并保存为wav文件。具体实现代码如下所示:

cpp 复制代码
#include <iostream>
#include <vector>
#include <complex>
#include <fftw3.h>
#include <alsa/asoundlib.h>
#include "math.h"
#include "stdio.h"
#include <vector>
#include <QDateTime>
#include <QDebug>
#include <fstream>
#include "sndfile.h"

using namespace std;
const int FRAME_SIZE = 1024;


// 定义采样率、通道数、采样格式等参数
#define SAMPLE_RATE 16000
#define CHANNELS 1
#define FORMAT SND_PCM_FORMAT_S16_LE // 16位,小端

int main() {
    // 初始化 ALSA 音频采集
    snd_pcm_t* capture_handle;
    int rc;

    rc = snd_pcm_open(&capture_handle, "default", SND_PCM_STREAM_CAPTURE, 0);
    if (rc < 0)
    {
        std::cerr << "无法打开默认音频设备: " << snd_strerror(rc) << std::endl;
            return 1;
    }

    // 设置音频采集参数
    // 配置PCM参数
    snd_pcm_hw_params_t *params;
    snd_pcm_hw_params_alloca(&params);
    snd_pcm_hw_params_any(capture_handle, params);
    snd_pcm_hw_params_set_access(capture_handle, params, SND_PCM_ACCESS_RW_INTERLEAVED);  //设置该参数的目的是决定多通道时是否采用一块内存来存储音频数据,设置SND_PCM_ACCESS_RW_INTERLEAVED参数就是将多通道情况下的数据存储到一块内存中
    snd_pcm_hw_params_set_format(capture_handle, params, FORMAT);
    snd_pcm_hw_params_set_channels(capture_handle, params, CHANNELS);
    unsigned int sampleRate = SAMPLE_RATE;
    snd_pcm_hw_params_set_rate_near(capture_handle, params, &sampleRate, 0);
    int err = snd_pcm_hw_params(capture_handle, params);
    if(err < 0)
    {
        fprintf(stderr, "无法打开 PCM 设备: %s\n", snd_strerror(err));
        return 1;
    }


    SNDFILE* sndfile;
    SF_INFO sfinfo;
    sfinfo.samplerate = 16000;
    sfinfo.channels = 1;
    sfinfo.format = SF_FORMAT_WAV | SF_FORMAT_PCM_16;
    sfinfo.frames = 0;
    sndfile = sf_open("output1.wav", SFM_WRITE, &sfinfo);
    if(!sndfile)
    {
        qDebug()<<"无法创建wav文件";
        snd_pcm_close(capture_handle);
        return 1;
    }
    int index = 0;
    short* audioFrame = (short*)malloc(FRAME_SIZE*sizeof(short));
    while (true)
    {

        rc = snd_pcm_readi(capture_handle, audioFrame, FRAME_SIZE);
        //vector<int16_t> vData(audioFrame, audioFrame+FRAME_SIZE);
        qDebug()<<"time :"<<QDateTime::currentMSecsSinceEpoch();
        if (rc == FRAME_SIZE)
        {

            sf_count_t ret = sf_write_short(sndfile, audioFrame, FRAME_SIZE);
            qDebug()<<"sf_write_short ret : "<<ret;
            index++;
            if(index == 50)
                break;

        }
        else if (rc < 0)
        {
            std::cerr << "读取音频数据时出错: " << snd_strerror(rc) << std::endl;
                break;
        }
    }
    sf_close(sndfile);
    snd_pcm_close(capture_handle);

    return 0;
}

该程序编译通过后,运行该程序,可以实现录音。

相关推荐
REDcker5 天前
WebCodecs VideoDecoder 的 hardwareAcceleration 使用
前端·音视频·实时音视频·直播·webcodecs·videodecoder
gihigo19985 天前
基于TCP协议实现视频采集与通信
网络协议·tcp/ip·音视频
山河君5 天前
四麦克风声源定位实战:基于 GCC-PHAT + 最小二乘法实现 DOA
算法·音视频·语音识别·信号处理·最小二乘法·tdoa
音视频牛哥5 天前
Android平台RTMP/RTSP超低延迟直播播放器开发详解——基于SmartMediaKit深度实践
android·人工智能·计算机视觉·音视频·rtmp播放器·安卓rtmp播放器·rtmp直播播放器
qq_416276425 天前
通用音频表征的对比学习
学习·音视频
美狐美颜sdk5 天前
Android全局美颜sdk实现方案详解
人工智能·音视频·美颜sdk·视频美颜sdk·美狐美颜sdk
EasyDSS5 天前
私有化部署EasyDSS视频点播能力:全链路视频技术赋能,打造企业级视听新体验
音视频·hls·m3u8·点播技术·智能转码
qq_416276425 天前
DeLoRes——一种通用的音频表征学习新方法(DeLoRes(基于 Barlow Twins 的冗余最小化方法)
学习·音视频
Q_4582838685 天前
从定位到视频:808 + 1078 在各行业的落地实践
音视频
山顶望月川5 天前
实测MiniMax-Hailuo-02:当“开工大吉“变成“无字天书“,国产AI视频模型的能与之不能
人工智能·音视频