【秒懂音视频开发】09_音频录制02_编程
通过编程录音
开发录音功能的主要步骤是:
- 注册设备
- 获取输入格式对象
- 打开设备
- 采集数据
- 释放资源

主要步骤
需要用到的FFmpeg库有3个。
复制代码
C++
extern "C" {
|
|
// 设备相关API
|
|
|
|
// 格式相关API
|
|
|
|
// 工具相关API(比如错误处理)
|
|
|
|
}
|
注册设备
在整个程序的运行过程中,只需要执行1次注册设备的代码。
复制代码
C++
// 初始化libavdevice并注册所有输入和输出设备
|
|
avdevice_register_all();
|
获取输入格式对象
宏定义
Windows和Mac环境的格式名称、设备名称都是不同的,所以使用条件编译实现跨平台。
复制代码
C++
// 格式名称、设备名称目前暂时使用宏定义固定死
|
|
|
|
// 格式名称
|
|
|
|
// 设备名称
|
|
|
|
|
|
|
|
|
|
|
核心代码
根据格式名称获取输入格式对象,后面需要利用输入格式对象打开设备。
复制代码
C++
AVInputFormat *fmt = av_find_input_format(FMT_NAME);
|
|
if (!fmt) {
|
|
// 如果找不到输入格式
|
|
qDebug() << "找不到输入格式" << FMT_NAME;
|
|
return;
|
|
}
|
打开设备
复制代码
C++
// 格式上下文(后面通过格式上下文操作设备)
|
|
AVFormatContext *ctx = nullptr;
|
|
// 打开设备
|
|
int ret = avformat_open_input(&ctx, DEVICE_NAME, fmt, nullptr);
|
|
// 如果打开设备失败
|
|
if (ret < 0) {
|
|
char errbuf[1024] = {0};
|
|
// 根据函数返回的错误码获取错误信息
|
|
av_strerror(code, errbuf, sizeof (errbuf));
|
|
qDebug() << "打开设备失败" << errbuf;
|
|
return;
|
|
}
|
采集数据
宏定义
复制代码
C++
|
|
// PCM文件的文件名
|
|
|
|
|
|
|
|
|
核心代码
复制代码
C++
|
|
// 文件
|
|
QFile file(FILENAME);
|
|
// WriteOnly:只写模式。如果文件不存在,就创建文件;如果文件存在,就删除文件内容
|
|
if (!file.open(QFile::WriteOnly)) {
|
|
qDebug() << "文件打开失败" << FILENAME;
|
|
// 关闭设备
|
|
avformat_close_input(&ctx);
|
|
return;
|
|
}
|
|
// 暂时假定只采集50个数据包
|
|
int count = 50;
|
|
// 数据包
|
|
AVPacket pkt;
|
|
// 从设备中采集数据
|
|
// av_read_frame返回值为0,代表采集数据成功
|
|
while (count-- > 0 && av_read_frame(ctx, &pkt) == 0) {
|
|
// 写入数据
|
|
file.write((const char *) pkt.data, pkt.size);
|
|
}
|
释放资源
复制代码
C++
// 关闭文件
|
|
file.close();
|
|
// 关闭设备
|
|
avformat_close_input(&ctx);
|
想要了解每一个函数的具体作用,可以查询:官方API文档。
多线程
录音属于耗时操作,为了避免阻塞主线程,最好在子线程中进行录音操作。这里创建了一个继承自QThread的线程类,线程一旦启动(start),就会自动调用run函数。
.h
复制代码
C++
|
|
class AudioThread : public QThread {
|
|
Q_OBJECT
|
|
private:
|
|
void run();
|
|
public:
|
|
explicit AudioThread(QObject *parent = nullptr);
|
|
~AudioThread();
|
|
};
|
.cpp
复制代码
C++
AudioThread::AudioThread(QObject *parent,
|
|
AVInputFormat *fmt,
|
|
const char *deviceName)
|
|
: QThread(parent), _fmt(fmt), _deviceName(deviceName) {
|
|
// 在线程结束时自动回收线程的内存
|
|
connect(this, &AudioThread::finished,
|
|
this, &AudioThread::deleteLater);
|
|
}
|
|
AudioThread::~AudioThread() {
|
|
// 线程对象的内存回收时,正常结束线程
|
|
requestInterruption();
|
|
quit();
|
|
wait();
|
|
}
|
|
void AudioThread::run() {
|
|
// 录音操作
|
|
// ...
|
|
}
|
开启线程
复制代码
C++
AudioThread *audioThread = new AudioThread(this);
|
|
audioThread->start();
|
结束线程
复制代码
C++
// 外部调用线程的requestInterruption,请求结束线程
|
|
audioThread->requestInterruption();
|
|
// 线程内部的逻辑
|
|
void AudioThread::run() {
|
|
// 可以通过isInterruptionRequested判断是否要结束线程
|
|
// 当调用过线程的requestInterruption时,isInterruptionRequested返回值就为true,否则为false
|
|
while (!isInterruptionRequested()) {
|
|
// ...
|
|
}
|
|
}
|
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· 上周热点回顾(2.24-3.2)