Qt源码-Qt多媒体音频框架

server/2024/10/9 15:21:57/

Qt 多媒体音频框架

  • 一、概述
  • 二、音频设计
    • 1. ALSA 基础
    • 2. Qt 音频类
      • 1. 接口实现
      • 2. alsa 插件实现

一、概述

环境详细
Qt版本Qt 5.15
操作系统Deepin v23
代码工具Visual Code
源码https://github.com/qt/qtmultimedia/tree/5.15

这里记录一下在Linux下Qt 的 Qt Multimedia 模块的设计,我目前先记录与音频相关的库的设计。不同Qt版本的设计有些不一样,需要看对应版本的源码设计。

二、音频设计

1. ALSA 基础

ALSA是 Advanced Linux Sound Architecture的缩写,即高级Linux声音架构。在Linux 2.6的内核版本后,ALSA目前已经成为了Linux的主流音频体系结构。
在这里插入图片描述

2. Qt 音频类

1. 接口实现

最重要的就是 QAudioInput和QAudioOutput两个类,作为音频输入输出的类。
在这里插入图片描述
这个类的源码位置在

src/multimedia/audio/qaudioinput.cpp

那我们开始看 QAudioInput 类的关键实现吧,在 qaudioinput.cpp 中都是使用的 qaudioinput.h 头文件中 QAbstractAudioInput 类型的 d 的成员方法。

如 QAudioInput::start() 实际使用的是 d->start();

// qaudioinput.h
private:QAbstractAudioInput* d;
// qaudioinput.cpp
void QAudioInput::start(QIODevice* device)
{d->start(device);
}
....
QAudioFormat QAudioInput::format() const
{return d->format();
}

在 QAudioInput 构造函数中,得知是d 是由工厂类 QAudioDeviceFactory 创建的

src/multimedia/audio/qaudiodevicefactory.cpp
QAudioInput::QAudioInput(const QAudioFormat &format, QObject *parent):QObject(parent)
{d = QAudioDeviceFactory::createDefaultInputDevice(format);...
}

QAudioDeviceFactory 工厂通过 createDefaultInputDevice() ⇒ \Rightarrow createInputDevice() 函数调用。所以具体就看 createInputDevice() 函数实现 就可以了。

QAbstractAudioInput* QAudioDeviceFactory::createDefaultInputDevice(QAudioFormat const &format)
{return createInputDevice(defaultDevice(QAudio::AudioInput), format);
}
QAbstractAudioInput* QAudioDeviceFactory::createInputDevice(QAudioDeviceInfo const& deviceInfo, QAudioFormat const &format)
{if (deviceInfo.isNull())return new QNullInputDevice();#if !defined (QT_NO_LIBRARY) && !defined(QT_NO_SETTINGS)QAudioSystemFactoryInterface* plugin =qobject_cast<QAudioSystemFactoryInterface*>(audioLoader()->instance(deviceInfo.realm()));if (plugin) {QAbstractAudioInput* p = plugin->createInput(deviceInfo.handle());if (p) p->setFormat(format);return p;}
#endifreturn new QNullInputDevice();
}

上面 createInputDevice 中 QAbstractAudioInput *p 就是 通过 audioLoader 加载的插件 plugin 创建出的。这样不同系统可以实现不同的插件,根据系统做到跨平台性。

QAudioSystemFactoryInterface* plugin 就是插件的接口父类。

在创建失败的时候,QAudioDeviceFactory 工厂也通过定义 QNullInputDevice 类。保证插件系统正常运行。

让我们看看QNullInputDevice 在 qaudiodevicefactory.cpp 的定义

class QNullInputDevice : public QAbstractAudioInput
{
public:void start(QIODevice*) override { qWarning()<<"using null input device, none available";}QIODevice *start() override { qWarning()<<"using null input device, none available"; return nullptr; void setBufferSize(int ) override {}int bufferSize() const override  { return 0; }
...void setVolume(qreal) override {}qreal volume() const override {return 1.0f;}
};

2. alsa 插件实现

那让我们继续看看音频功能的具体实现,音频插件的源码工程位置在 `

src/pligins/alsa/alsa.pro

在下面我们可以看到,不同平台的不同实现,有Linux、Windows、Android等。在我WIndows电脑的Qt5.14.2 上音频库则是使用的
windowsaudio 工程
在这里插入图片描述

我们找到Linux的alsa工程,从工程文件可以得到这个插件的名字叫 qtaudio_alsa
在这里插入图片描述
这个插件最后生成的名字叫 libqtaudio_alsa.so

在我的系统中安装的位置在

/usr/lib/x86_64-linux-gnu/qt5/plugins/audio/libqtaudio_alsa.so	

在这里插入图片描述

下面继续介绍这个插件

qalsaplugin.cpp:就是生成插件的接口类,对外使用,内部使用了ALSA实现功能

qalsaaudiodeviceinfo.cpp:封装的 音频设备信息,功能QAudioDeviceInfo 功能
qalsaaudioinput.cpp :音频输入类,实际 QAudioInput 的实现
qalsaaudiooutput.cpp:音频输出类,实际的 QAudioOutput 的实现

下面就是 qalsaaudioinput 的open函数实现,可以看到open函数使用了 snd_pcm_open 函数:

bool QAlsaAudioDeviceInfo::open()
{int err = 0;QString dev;if (!availableDevices(mode).contains(device.toLocal8Bit()))return false;#if SND_LIB_VERSION < 0x1000e  // 1.0.14if (device.compare(QLatin1String("default")) != 0)dev = deviceFromCardName(device);else
#endifdev = device;if(mode == QAudio::AudioOutput) {err=snd_pcm_open( &handle,dev.toLocal8Bit().constData(),SND_PCM_STREAM_PLAYBACK,0);} else {err=snd_pcm_open( &handle,dev.toLocal8Bit().constData(),SND_PCM_STREAM_CAPTURE,0);}if(err < 0) {handle = 0;return false;}return true;
}

http://www.ppmy.cn/server/128762.html

相关文章

雷池+frp 批量设置proxy_protocol实现真实IP透传

需求 内网部署safeline&#xff0c;通过frp让外网访问内部web网站服务&#xff0c;让safeline记录真实外网攻击IP safeline 跟 frp都部署在同一台服务器&#xff1a;192.168.2.103 frp client 配置 frpc只需要在https上添加transport.proxyProtocolVersion "v2"即…

MySQL基础篇 - 函数

01 函数的介绍 【1】之前学习的聚合函数只是函数中的一种。 【2】函数&#xff1a;是一段可以直接被另一段程序调用的程序/代码。 【3】应用场景示例&#xff1a; ① 已知某员工的入职信息&#xff0c;求员工入职天数。 ② 已某同学成绩&#xff0c;求它的成绩等级。 【4】…

Redis中数据类型的使用(hash和list)

&#xff08;一&#xff09;hash哈希 我们知道redis中的数据都是以键值对的方式存储的&#xff0c;key全部都是string类型&#xff0c;而value可以是不同的数据结构&#xff0c;其中就包括hash&#xff0c;也就是说&#xff0c;key这一层组织完成后到了value仍然是hash 1.Hash…

Elasticsearch基础_2.数据类型

文章目录 一、基本的数据类型1.1、keyword1.2、text1.3、数值类型1.4、布尔类型1.5、时间类型 二、复杂的数据类型三、字段映射 一、基本的数据类型 1.1、keyword keyword类型是不进行切分的字符串类型。这里的“不进行切分”指的是&#xff1a;在索引时&#xff0c;对keyword…

【CSS】水平垂直居中

给父盒子设置属性 flex display: flex;写在父元素上这就是定义了一个伸缩容器justify-content&#xff1a;center 设置主轴对齐方式为居中&#xff0c;默认是横轴。子元素居中。align-items&#xff1a;center 设置纵轴对齐方式为居中&#xff0c;默认是纵轴。子元素居中。 给…

C语言期中自测试卷

选择题 1、若有变量定义int a; double b; 要输入数据存放在a和b中&#xff0c;则下面正确的输入数据的语句为&#xff1a; 【 正确答案: C】 A. scanf("%d%f",a,b); B. scanf("%d%f",&a,&b); C. scanf("%d%lf",&a,&b); D. scan…

【设计模式-模板】

定义 模板方法模式是一种行为设计模式&#xff0c;它在一个方法中定义了一个算法的骨架&#xff0c;并将一些步骤延迟到子类中实现。通过这种方式&#xff0c;模板方法允许子类在不改变算法结构的情况下重新定义算法中的某些特定步骤。 UML图 组成角色 AbstractClass&#x…

Object.values() 、 Object.keys()

拿到当前对象里面的value值 // 假设你有一个对象 const myObject {name: Kimi,age: 30,country: Moon };// 获取对象的所有值 const values Object.values(myObject);// 输出值数组 console.log(values); // ["Kimi", 30, "Moon"] 如果你需要在 Vue 组…