Qt/QML编程学习之心得:D-BUS进程间通信(四)

news2025/1/10 23:32:09

Qt/QML应用编程最适合于一些触摸的嵌入式界面设计,那么GUI界面怎么与底层的设备通信,怎么与一个系统内其他模块通信的呢?这就不得不说一个很重要的设计模式:d-bus

D-BUS是一个系统中消息总线,用于IPC/RPC。消息系统很简单而功能强大,可以在一些命令行实用程序的帮助下进行操作(dbus-send)-以及一些GUI工具(qdbusviewer)。

D-BUS是一个进程间通信(IPC)系统,提供了一种简单而强大的机制允许应用程序彼此交谈、通信信息和请求服务。D-BUS是从头开始设计,以满足现代Linux系统的需求。D-BUS的最初目标是成为分别用于GNOME和KDE的远程对象系统CORBA和DCOP的替代。理想情况下,D-BUS可以成为两台台式机使用的统一且不可知的IPC机制,满足它们需求和引入新功能。D-BUS作为一个功能齐全的工控机和对象系统,有几个预期用途。首先,D-BUS可以执行基本功能应用程序IPC,允许一个进程将数据传送到另一个进程。想想类固醇上的UNIX域套接字。其次,D-BUS可以方便地通过系统发送事件或信号,允许不同的组件在系统中进行通信并最终更好地集成。例如,蓝牙dæmon可以发送音乐播放器可以截获的来电信号,在通话结束前将音量静音。最后D-BUS实现了一个远程对象系统,允许一个应用程序从另一个对象认为CORBA没有复杂性。

D-Bus真正的强大之处在于它丰富的编程接口。D-Bus本身是在C中实现的,因此低级别的C API是D-Bus的“自然”API。然而,事实确实如此使用起来相当麻烦,因此存在用于D总线的各种绑定。

为何D-Bus是独特的?Why D-BUS Is Unique

D-BUS在几个方面与其他IPC机制不同。首先,D-BUS中IPC的基本单元是消息,而不是字节流。通过这种方式,D-BUS将IPC分解为离散的消息,包括报头(元数据)和有效载荷(数据)。消息格式为二进制、类型化、完全对齐且简单。它是有线协议的固有部分。这种方法与其他IPC机制形成对比,其中franca是一个随机的字节流,而不是一个离散的消息。其次,D-BUS是基于总线的。最简单的沟通方式是过程对过程。然而D-BUS,提供了一个dæmon,称为消息总线守护进程,用于在特定的进程之间路由消息总线通过这种方式,形成了总线拓扑,允许进程在同时处理一个或多个应用。应用程序可以向总线发送或侦听总线上的各种事件。最后一个独特的功能是创建两条总线,而不是一条,即系统总线和会话总线。系统总线是全局的、系统范围的,并且在系统级别运行。系统的所有用户都可以进行通信通过具有适当权限的总线,允许系统范围事件的概念。会话总线,但是,它是在用户登录期间创建的,并在用户或会话级别运行。此总线仅由特定用户,在特定登录会话中,作为用户应用程序的IPC和远程对象系统。

D-BUS概念

消息被发送到对象。对象使用路径名进行寻址,例如.org/cups/printers/queue。消息总线上的进程与对象相关联,并在该对象上实现接口。D-BUS支持多种消息类型,如信号、方法调用、方法返回和错误消息。信号是特定事件发生的通知。它们是简单的、异步的、单向的抬头信息。方法调用消息允许应用程序请求调用远程上的方法对象方法返回消息提供方法调用产生的返回值。错误消息提供异常以响应方法调用。D-BUS是完全类型化和类型安全的。消息的标头和有效负载都是完全类型化的。有效类型包括字节,布尔值,32位整数,32位无符号整数,64位整数,64比特无符号整数,双精度浮点和字符串。特殊的数组类型允许对类型进行分组。格言类型允许使用字典样式的键/值对。D-BUS是安全的。它实现了一个基于SASL配置文件的简单协议,用于一对一身份验证连接。在总线范围内,从特定接口读取和写入消息是由安全系统控制。管理员可以控制对总线上任何接口的访问。D总线dæmon从头开始写的时候就考虑到了安全性。

为何D-BUS?

这些概念很有说服力,但有什么好处呢?首先,全系统消息总线是一个新概念。整个系统共享的单个总线允许从内核传播事件到系统上最上面的应用程序。Linux,具有定义良好的接口和层的清晰分离,不是很集成。D-BUS的系统消息总线改进了集成,而无需损害了良好的工程实践。现在,磁盘已满和打印机队列为空等事件甚至电池电量低可能会使系统堆栈膨胀,可用于任何需要的应用程序,从而允许系统进行响应和反应。事件是异步发送的,并且不进行轮询。

内核事件层。

内核事件层是一种使用高速网络链接套接字的内核到用户通信机制以与用户空间异步通信。该机制可以连接到D-BUS,允许内核发送D-BUS信号!内核事件层与sysfs绑定,sysfs是现代Linux系统上位于/sys的kobjects树。每个sysfs中的目录与kobject绑定,kobject是内核中用于表示对象的结构;sysfs是一个导出为文件系统的对象层次结构。每个内核事件层事件都被建模为好像它源自一个sysfs路径。因此,事件出现了就好像它们从kobjects发射一样。sysfs路径很容易转换为D-BUS路径,使内核事件层和D-BUS自然搭配。这个内核事件层被合并到2.6.10-rc1内核中。其次,会话总线为IPC和远程方法调用提供了一种机制,可能会提供GNOME和KDE之间的统一系统。D-BUS旨在成为一个比CORBA更好的CORBADCOP而非DCOP,在提供附加功能的同时满足了两个项目的需求。而且,D-BUS在保持简单高效的同时完成了所有这些。

如何在应用中增加D-Bus?

核心的D-BUS API是用C语言编写的,是相当低级和大型的。在此API之上,绑定与编程语言和环境,包括Glib、Python、Qt和Mono。除了提供语言包装器,绑定提供特定于环境的功能。例如,Glib绑定处理D-BUS连接作为GObjects,并允许消息传递集成到Glib主循环中。首选用途D-BUS肯定在使用特定于语言和环境的绑定,这既是为了方便使用,也是为了改进功能。

Qt中,要使用QtDBus模块,需要#include <QtDBus>,.pro文件中Qt+=dbus,QtDBus模块通过QDBusArgument类实现了类型系统,允许用户通过总线发送和接收每一种C++类型。

当需要多对多通信时,使用Dbus。为了实现这一点,在任何应用程序连接到总线之前,都会启动一个中央服务器:该服务器负责跟踪连接的应用程序,并将消息从源正确路由到目的地。此外,D-Bus定义了两个众所周知的总线,称为系统总线和会话总线。这些总线的特殊之处在于它们具有定义明确的语义:一些服务被定义为在其中一条或两条总线中找到。例如,希望查询连接到计算机的硬件设备列表的应用程序可能会与系统总线上可用的服务通信,而提供打开用户网络浏览器的服务可能会在会话总线上找到。在系统总线上,人们还可以期望发现每个应用程序可以提供哪些服务的限制。因此,可以合理地确定,如果某个服务存在,那么它是由受信任的应用程序提供的。

Messages信息

在底层,应用程序通过Dbus相互发送消息进行通信。消息用于中继远程过程调用以及与之相关的回复和错误。当在bus上使用时,消息有一个目的地,这意味着它们只发送给感兴趣的各方,避免了由于“拥挤”或广播而造成的拥堵。然而,一种被称为“Signal message信号消息”的特殊消息(基于Qt的信号和插槽机制的概念)没有预定义的目的地。由于其目的是在一对多上下文中使用,因此信号消息被设计为通过“选择加入”机制工作。QtD-Bus模块将消息的低级概念完全封装到Qt开发人员熟悉的更简单、面向对象的方法中。在大多数情况下,开发人员不必担心发送或接收消息。

Services Name服务名称

当通过总线进行通信时,应用程序会获得所谓的“服务名称”:即该应用程序如何选择被同一总线上的其他应用程序所知。服务名称由D-Bus总线守护进程代理,用于将消息从一个应用程序路由到另一个应用。与服务名称类似的概念是IP地址和主机名:根据计算机向网络提供的服务,计算机通常有一个IP地址,并且可能有一个或多个与其相关的主机名。另一方面,如果不使用总线,也不会使用服务名称。如果我们再次将其与计算机网络进行比较,这将等同于点对点网络:由于对等端是已知的,因此不需要使用主机名来查找它或其IP地址。D-Bus服务名称的格式实际上与主机名非常相似:它是一个由字母和数字点分隔的序列。通常的做法甚至是根据定义服务的组织的域名来命名服务名称。比如org.freedesktop.DBus

Object Path对象路径

与网络主机一样,应用程序通过导出对象向其他应用程序提供特定服务。这些对象是分层组织的,很像从QObject派生的类所拥有的父子关系。然而,一个不同之处在于,存在“根对象”的概念,即所有对象都有最终的父对象。如果我们继续与Web服务进行类比,对象路径等同于URL的路径部分:

 与它们一样,D-Bus中的对象路径的形式类似于文件系统上的路径名:它们是斜杠分隔的标签,每个标签由字母、数字和下划线(“_”)组成。它们必须始终以斜线开头,而不能以斜线结尾。

Interface接口

接口类似于C++抽象类和Java的interface关键字,并声明调用者和被调用者之间建立的“契约”。也就是说,它们建立了可用的方法method、信号signal和属性property的名称,以及建立通信时双方期望的行为。Qt在其插件系统中使用了一种非常类似的机制:C++中的基类通过Q_DECLARE_INTERFACE()宏与一个唯一的标识符相关联。事实上,D-Bus接口名称的命名方式与Qt插件系统所建议的类似:一个通常由定义该接口的实体的域名构建的标识符。

 Dbus支持3种复合类型:ARRAY、STRUCT和 maps/dictionaries,两种非原生类型:QStringList和QByteArray,还有BYTE、INY16等原生类型。如果自定义类性须使用Q_DECLARE_METATYPE()声明为Qt元类型,使用qDBusRegisterMetaType()函数注册,流操作符会被注册系统自动找到。

QtDBus的几个常用类

QDBusMessage类:D-Bus总线发送或接收的一个消息,有MethodCallMessage、 SignalMessage、 ReplyMessage、 ErrorMessage等

QDBusConnection类:到D-Bus总线的一个连接,是一个D-Bus会话的起始点。通过QDBusConnection连接对象,可以访问远程对象、接口,连接远程信号到本地槽函数,注册对象等。connectToBus()函数会创建一个到总线服务端的连接,QDBusConnection::send()函数发送消息。QDBusPendingCall asyncCall(const QDBusMessage & message, int timeout = -1)const返回一个打开到system总线的QDBusConnection对象,QDBusMessage call(const QDBusMessage & message, QDBus::CallMode mode = QDBus::Block, int timeout = -1 ) const发送message消息到连接,并立即返回,bool registerService(const QString & serviceName)注册object对象到路径path。

QDBusInterface远程对象接口的代理,连接到远程对象导出的信号,获取/设置远程属性的值。QObject::connect()连接,调用QObject::property()、QObject::setProperty()对属性进行访问。

QDBusReply存储对远程对象的方法调用的应答,是方法调用的应答QDBusMessage对象的一个子集.

QDBusAbstractAdaptor用于使用D-Bus向外部提供接口,使用信号、槽、属性来决定哪些被暴露到bus总线。

QDBusAbstractInterface允许访问远程接口的所有D-Bus接口的基类。

QDBusConnectionInterface访问D-Bus总线服务。

 最早的d-bus来自于redhat linux的一个项目。

 Inter-Process Communication (IPC)就是进程间通信。

 使用D-Bus的IPC :

 How D-bus is working?

 system bus有点像system service系统服务,使用的话直接调用就好了。而session bus会话总线也是类似service的方式提供接口服务。

下图展现了D-Bus的组成部分:

 D-Bus/service服务:

 D-Bus/Objects对象:

 D-Bus/Interfaces接口:

 

 D-Bus/Properties属性:

 D-Bus/Method方法:

 D-Bus/signal信号:

 D-Bus/Policy策略:

 D-Bus/Libraries & bindings

 

 Tools工具:

 

 

 

 示例:

 

 

 

 结论:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/436634.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

vi编辑器的使用介绍

vi编辑器的使用 vi的特点与运用场景vi的使用简易执行一个案例按键说明第一部分&#xff1a;命令模式的按键说明(光标移动、复制粘贴、查找替换)移动光标的方法查找与替换删除、复制与粘贴 第二部分&#xff1a;命令模式切换到输入模式的可以按键进入插入或替换的编辑模式 第三部…

A100 Jeston TX1/TX2使用教程-介绍

大家好&#xff0c;我是虎哥&#xff0c;经过一段时间的整理&#xff0c;终于完成了我自己算力盒子&#xff0c;A100系统的设计和研发&#xff0c;今天就来和大家聊聊这款针对TX1和TX2的入门级计算盒子的一些特性和功能。 一、EdgeBox_Umate_A100 算力盒子 A100 算力盒子是“玩…

系统集成项目管理工程师 笔记(第五章:项目立项管理)

文章目录 5.1 项目建议 2225.2 项目可行性分析 224项目可行性研究内容&#xff1a;5.2.2 项目可行性研究阶段 227 5.4 项目招投标 229《中华人民共和国招标投标法实施条例》5.4.1 项目招标 2295.4.2 项目投标 2305.4.3 开标与评标 2345.4.4 选定项目承建方 235 5.5 项目合同谈判…

实模式下内存访问

虽然有了寄存器&#xff0c;但是数据和指令还是需要存储到内存中。通常情况下需要把数据从内存中放到寄存器中才能使用&#xff0c;同样的指令需要放到寄存器中才能被CPU执行。 所有的内存访问都需要段寄存器左移四位加上其他寄存器的值才能得到真正地址值。这是由于以前运行实…

Unity使用ShaderGragh制作透明指针

Unity使用ShaderGragh制作透明指针 1 概述2 使用环境3 制作流程3.1 创建一个ShaderGragh3.2 打开ShaderGraph编辑器3.3 编辑器界面介绍3.4 Shader节点和部分信息如下3.5 常用节点介绍3.6 使用Shader3.7 贴图规范 4 控制Shader旋转4.1 API介绍4.2 示例代码&#xff1a;3.9 Shade…

Redis 6.0+ 的 ACL 机制

目录 前言一、安装 Redis 服务二、创建 ACL 用户三、用户密码管理3.1 删除密码3.2 重置用户和密码 四、权限管理4.1 key 管理4.2 权限管理 五、ACL 用户存储5.1 配置文件实现5.2 外部 ACL 文件实现 前言 Redis 6.0 引入了 ACL 机制&#xff0c;类似 MySQL 一样全部权限管理&am…

表单验证:自定义校验规则

Element UI 为我们提供了表单校验规则&#xff0c;但业务需要&#xff0c;我们常常要自定义校验规则 需求 实现表单中一个输入框&#xff0c;不能输入大于30的数字 思路 hrml&#xff1a; 自定义校验规则&#xff1a; 约定的校验规则&#xff1a; 代码 <template&g…

集群聊天服务器项目(一)——模块分层设计

本项目对程序不同功能进行分层设计&#xff0c;分为网络层、业务层、和数据层。 C面向接口编程也就是面向抽象类&#xff0c;网络模块和业务模块尽量解耦。 网络层 网络层主要封装的是网络连接方面的一些功能&#xff0c;即socket相关操作,这里该项目采用的是muduo网络库作为…

《Netty》从零开始学netty源码(三十九)之PoolSubPage的内存释放

PoolSubPage.free PoolSubPage的内存释放相对来说比较简单&#xff1a; 首先根据段的偏移量bitmapIdx找到bitmap的long[]数组的索引q&#xff0c;将bitmap[q]这个long的二进制位的占用位r置为0&#xff0c;表示已经释放。如果PoolSubPage的段已经全部释放了&#xff0c;且池中…

测试开发岗 - 常见面试题

1. 什么是软件测试&#xff0c; 谈谈你对软件测试的了解 软件测试就是验证产品特性是否符合用户需求, 软件测试贯穿于软件的整个生命周期. >>> 那软件测试具体是什么呢 ? 就拿生活中的例子来说, 比如说我们去商场买衣服, 会有以下几个步骤 : 第一步: 我们会走进门店…

【网络安全】命令执行漏洞

命令执行漏洞 命令执行漏洞原理危害检测方法有回显检测方法; (分号) 从左到右执行| (管道符) 将见面命令的输入为后面命令的标准输入&(后台任务符号) 命令从左到右执行&&(与) 逻辑与&#xff0c;前面命令执行成功后才会执行||(或) 逻辑或&#xff0c;前面执行失败才…

LeetCode算法小抄-- 图的遍历

LeetCode算法小抄-- 图的遍历 图基本概念遍历广度优先算法(BFS)框架[111. 二叉树的最小深度](https://leetcode.cn/problems/minimum-depth-of-binary-tree/)[752. 打开转盘锁](https://leetcode.cn/problems/open-the-lock/)[773. 滑动谜题](https://leetcode.cn/problems/sli…

文章伪原创生成器在线-文章伪原创工具免费入口

文章自动生成器 在现代科技快速发展的时代中&#xff0c;自动化技术已经深入到了各个领域。而随着人工智能技术的提高&#xff0c;自动化技术在创意和写作领域越来越成熟。现在有一款名为“文章自动生成器”的软件&#xff0c;可以轻松地生成高质量的文章。 今天&#xff0c;我…

STM32之MPU6050获取欧拉角

STM32之MPU6050获取欧拉角 MPU6050MPU6050特点MPU6050电路图以及框图MPU6050框图MPU6050电路图 MPU6050相关寄存器电源管理寄存器1&#xff08;0x6B&#xff09;陀螺仪配置寄存器&#xff08;0x1B&#xff09;加速度计配置寄存器&#xff08;0x1C&#xff09;陀螺仪采样率分频寄…

Vue中的ajax【Vue】

4. Vue 中的 ajax 4.1 解决开发环境 Ajax 跨域问题 方法一&#xff1a; 在vue.config.js中添加如下配置&#xff1a; devServer:{proxy:"http://localhost:5000" }说明&#xff1a; 优点&#xff1a;配置简单&#xff0c;请求资源时直接发给前端&#xff08;808…

更懂业务的用友iuap平台,助力企业升级数智化底座

4月19日&#xff0c;一年一度的用友BIP技术大会如约而至。近千位来自三十个行业的企业家、CIO/CDO、企业主管、专家学者、媒体、分析师代表现场参与大会。伴随企业数智化推进&#xff0c;越来越多的企业需要升级数智底座平台。会上&#xff0c;用友介绍了更懂企业业务的用友BIP…

Android 开发为什么会要用到组件化与插件化?好处在哪?

对于开发者来说&#xff0c;写好代码的第一步就是具备良好的架构能力。但是这项基本的能力&#xff0c;也很少有人具备。就拿最常用的项目架构组件化来说&#xff0c;有多少人用过&#xff1f;又有谁去了解过组件化开发中真正会遇到的问题&#xff0c;以及如何解决&#xff1f;…

Nacos 1.4.x 升级至 2.x 详细步骤及遇到的问题,亲测可行

此前使用的nacos版本是1.4.5&#xff0c;现在nacos最新版本为2.2.2&#xff0c;且修复了旧版本的一些安全问题&#xff0c;下面把详细的升级步骤记录一下&#xff0c;大家一起学习&#xff0c;亲测有效。 主要参考nacos官方升级文档&#xff1a;https://nacos.io/zh-cn/docs/v2…

瑞吉外卖项目——读写分离

读写分离 读和写所有压力都由一台数据库承担&#xff0c;压力大数据库服务器磁盘损坏则数据丢失&#xff0c;单点故障 Mysql主从复制 介绍 MySQL主从复制是一个异步的复制过程&#xff0c;底层是基于Nysql数据库自带的二进制日志功能。 就是一台或多台MysQL数据库&#xf…

字符串 --- 找子串匹配算法

1.基本介绍 主串&#xff1a;形如 “hello world”的字符串作为一个整体 子串&#xff1a;上面主串的一部分如“world” 在计算机世界&#xff0c;主串找子串的模式很常见&#xff0c;比如要在word文件中找一句指定的话&#xff0c;那么面对海量的信息&#xff0c;我们匹配算法…