Qt 监控USB设备的插入和移除
Qt 监控USB设备的插入和移除
flyfish
Ubuntu22.04
Qt 6.2.4
CMakeLists.txt
内容
# 指定 CMake 的最低版本要求
cmake_minimum_required(VERSION 3.16)# 定义项目的名称和使用的编程语言
project(USBMonitor LANGUAGES CXX)# 开启自动 UIC,MOC 和 RCC 工具
set(CMAKE_AUTOUIC ON) # 自动运行 uic 工具处理 .ui 文件
set(CMAKE_AUTOMOC ON) # 自动运行 moc 工具处理 Qt 的元对象系统
set(CMAKE_AUTORCC ON) # 自动运行 rcc 工具处理资源文件# 设置 C++ 标准为 C++17,并且要求编译器必须支持 C++17
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_STANDARD_REQUIRED ON)# 查找 Qt 库,并指定所需的组件
find_package(QT NAMES Qt6 Qt5 REQUIRED COMPONENTS Core libudev) # 尝试查找 Qt6 或 Qt5 库,并且要求 Core 和 libudev 组件
find_package(Qt${QT_VERSION_MAJOR} REQUIRED COMPONENTS Core) # 根据找到的 Qt 版本(Qt6 或 Qt5),再次查找 Core 组件# 包含 GNUInstallDirs 模块,该模块定义了一些标准的安装目录变量
include(GNUInstallDirs)# 定义一个可执行目标 USBMonitor,并指定其源文件为 main.cpp
add_executable(USBMonitormain.cpp
)# 将 Qt Core 库和 udev 库链接到 USBMonitor 可执行文件
target_link_libraries(USBMonitor PRIVATE Qt${QT_VERSION_MAJOR}::Core udev)# 定义安装目标 USBMonitor 的安装路径
install(TARGETS USBMonitorLIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR} # 指定库文件的安装路径RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR} # 指定可执行文件的安装路径
)
编译过程中的内部操作
-
解析
CMakeLists.txt
:- CMake 读取
CMakeLists.txt
文件,解析其中的指令和变量设置。
- CMake 读取
-
查找和配置依赖库:
find_package
命令会查找 Qt 库和libudev
库,并设置相应的变量,如Qt6::Core
和UDEV_LIBRARIES
。
-
生成构建文件:
- CMake 生成适合特定构建系统的文件,如 Makefile(对于 Unix 系统)或 Visual Studio 项目文件(对于 Windows 系统)。
-
编译源文件:
- 构建系统(如
make
)根据生成的构建文件编译源文件main.cpp
,生成目标文件main.cpp.o
。
- 构建系统(如
-
链接目标文件:
- 构建系统将目标文件
main.cpp.o
与Qt6::Core
和udev
库链接,生成最终的可执行文件USBMonitor
。
- 构建系统将目标文件
-
安装目标(可选):
- 如果运行
make install
命令,构建系统会将生成的可执行文件USBMonitor
安装到指定的目录,如/usr/local/bin
。
- 如果运行
源码
#include <QCoreApplication> // 包含 Qt 核心模块
#include <QSocketNotifier> // 包含 QSocketNotifier 类
#include <QDebug> // 包含调试输出功能
#include <unistd.h> // 包含 POSIX 操作系统 API
#include <fcntl.h> // 包含文件控制选项
#include <libudev.h> // 包含 libudev 库
#include <sys/inotify.h> // 包含 inotify API
#include <errno.h> // 包含错误码定义// 定义 UsbMonitor 类,继承自 QObject
class UsbMonitor : public QObject {Q_OBJECT // 宏,用于 Qt 元对象系统public:// 构造函数UsbMonitor(QObject *parent = nullptr) : QObject(parent) {// 初始化 udevudev = udev_new(); // 创建 udev 上下文if (!udev) {qCritical() << "无法初始化 udev"; // 如果初始化失败,输出错误信息return;}// 创建 udev 监视器udev_monitor = udev_monitor_new_from_netlink(udev, "udev"); // 创建 udev 监视器if (!udev_monitor) {qCritical() << "无法创建 udev 监视器"; // 如果创建失败,输出错误信息return;}// 过滤只监听 USB 设备udev_monitor_filter_add_match_subsystem_devtype(udev_monitor, "usb", NULL); // 过滤只监听 USB 子系统的设备udev_monitor_enable_receiving(udev_monitor); // 启用接收事件// 获取 udev 监视器的文件描述符int udev_fd = udev_monitor_get_fd(udev_monitor); // 获取 udev 监视器的文件描述符// 检查文件描述符是否有效if (fcntl(udev_fd, F_GETFD) == -1) { // 使用 fcntl 检查文件描述符是否有效qCritical() << "无效的文件描述符:" << udev_fd; // 如果无效,输出错误信息return;}// 创建 QSocketNotifier 并连接信号notifier = new QSocketNotifier(udev_fd, QSocketNotifier::Read, this); // 创建 QSocketNotifier 对象connect(notifier, &QSocketNotifier::activated, this, &UsbMonitor::onUdevEvent); // 连接 QSocketNotifier 的 activated 信号到 onUdevEvent 槽函数}// 析构函数~UsbMonitor() {if (notifier) {delete notifier; // 释放 QSocketNotifier 对象}if (udev_monitor) {udev_monitor_unref(udev_monitor); // 释放 udev 监视器}if (udev) {udev_unref(udev); // 释放 udev 上下文}}private slots:// 处理 udev 事件的槽函数void onUdevEvent(int socket) {// 从 udev 监视器读取事件struct udev_device *dev = udev_monitor_receive_device(udev_monitor); // 从 udev 监视器读取设备事件if (dev) {const char *action = udev_device_get_action(dev); // 获取事件动作const char *devnode = udev_device_get_devnode(dev); // 获取设备节点const char *devtype = udev_device_get_devtype(dev); // 获取设备类型if (action && devnode && devtype) {if (strcmp(action, "add") == 0) {qDebug() << "USB 设备插入:" << devnode; // 如果是插入事件,输出设备节点} else if (strcmp(action, "remove") == 0) {qDebug() << "USB 设备移除:" << devnode; // 如果是移除事件,输出设备节点}}udev_device_unref(dev); // 释放 udev_device 对象}}private:struct udev *udev; // udev 上下文struct udev_monitor *udev_monitor; // udev 监视器QSocketNotifier *notifier; // QSocketNotifier 对象
};// 主函数
int main(int argc, char *argv[]) {QCoreApplication a(argc, argv); // 创建 QCoreApplication 对象UsbMonitor monitor; // 创建 UsbMonitor 对象return a.exec(); // 进入事件循环
}#include "main.moc" // 包含 moc 生成的代码
运行结果
USB 设备移除: /dev/bus/usb/002/006
USB 设备插入: /dev/bus/usb/002/007
USB 设备移除: /dev/bus/usb/002/007
USB 设备插入: /dev/bus/usb/002/008
#include "main.moc"
的主要作用是确保 moc(Meta-Object Compiler)生成的代码能够被编译器正确处理。具体来说,moc 会生成一些额外的代码,这些代码需要被包含在源文件的末尾,以便在编译时能够正确链接和使用。
Qt 元对象系统:
Qt 的元对象系统提供了信号和槽机制、运行时类型信息(RTTI)、动态属性系统等功能。
为了实现这些功能,Qt 提供了一个工具 moc(Meta-Object Compiler),它会生成一些额外的 C++ 代码。
moc 生成的代码:
当在类中使用 Q_OBJECT 宏时,moc 会为该类生成一些额外的代码,这些代码包括信号和槽的实现、元对象数据等。
生成的代码通常保存在一个与源文件同名的 .moc 文件中,例如 main.cpp 对应的 main.moc。
包含 main.moc:
为了确保生成的代码能够被编译器正确处理,需要在源文件的末尾包含 main.moc 文件。
这样做是为了确保 moc 生成的代码在编译时能够被正确链接到的源文件中。
编译顺序:
编译器在处理源文件时,会按照文件中出现的顺序依次处理每一行代码。
如果在源文件的开头包含 main.moc,可能会导致编译器在处理 moc 生成的代码时,还没有看到类的定义,从而引发编译错误。
因此,通常在源文件的末尾包含 main.moc,确保类的定义已经完全可见。
libudev
库提供的函数 ,用于管理和监控硬件设备的变化。
1. udev_new()
- 原型:
struct udev *udev_new(void);
- 作用:创建一个新的
udev
上下文。 - 返回值:成功时返回指向
udev
结构的指针,失败时返回NULL
。 - 示例:
udev = udev_new(); if (!udev) {qCritical() << "无法初始化 udev";return; }
2. udev_monitor_new_from_netlink()
- 原型:
struct udev_monitor *udev_monitor_new_from_netlink(struct udev *udev, const char *name);
- 作用:创建一个新的
udev
监视器,用于监听来自内核的netlink
事件。 - 参数:
udev
:udev
上下文。name
:监视器的名称,通常为"udev"
。
- 返回值:成功时返回指向
udev_monitor
结构的指针,失败时返回NULL
。 - 示例:
udev_monitor = udev_monitor_new_from_netlink(udev, "udev"); if (!udev_monitor) {qCritical() << "无法创建 udev 监视器";return; }
3. udev_monitor_filter_add_match_subsystem_devtype()
- 原型:
int udev_monitor_filter_add_match_subsystem_devtype(struct udev_monitor *udev_monitor, const char *subsystem, const char *devtype);
- 作用:为
udev
监视器添加一个过滤器,只监听特定子系统和设备类型的事件。 - 参数:
udev_monitor
:udev
监视器。subsystem
:要监听的子系统,例如"usb"
。devtype
:要监听的设备类型,可以为NULL
表示匹配所有设备类型。
- 返回值:成功时返回 0,失败时返回负数。
- 示例:
udev_monitor_filter_add_match_subsystem_devtype(udev_monitor, "usb", NULL);
4. udev_monitor_enable_receiving()
- 原型:
int udev_monitor_enable_receiving(struct udev_monitor *udev_monitor);
- 作用:启用
udev
监视器,开始接收事件。 - 参数:
udev_monitor
:udev
监视器。
- 返回值:成功时返回 0,失败时返回负数。
- 示例:
udev_monitor_enable_receiving(udev_monitor);
5. udev_monitor_get_fd()
- 原型:
int udev_monitor_get_fd(struct udev_monitor *udev_monitor);
- 作用:获取
udev
监视器的文件描述符,用于监听事件。 - 参数:
udev_monitor
:udev
监视器。
- 返回值:成功时返回文件描述符,失败时返回负数。
- 示例:
int udev_fd = udev_monitor_get_fd(udev_monitor);
6. udev_monitor_receive_device()
- 原型:
struct udev_device *udev_monitor_receive_device(struct udev_monitor *udev_monitor);
- 作用:从
udev
监视器中读取下一个设备事件。 - 参数:
udev_monitor
:udev
监视器。
- 返回值:成功时返回指向
udev_device
结构的指针,失败时返回NULL
。 - 示例:
struct udev_device *dev = udev_monitor_receive_device(udev_monitor);
7. udev_device_get_action()
- 原型:
const char *udev_device_get_action(struct udev_device *udev_device);
- 作用:获取设备事件的动作(例如
"add"
或"remove"
)。 - 参数:
udev_device
:设备对象。
- 返回值:成功时返回动作字符串,失败时返回
NULL
。 - 示例:
const char *action = udev_device_get_action(dev);
8. udev_device_get_devnode()
- 原型:
const char *udev_device_get_devnode(struct udev_device *udev_device);
- 作用:获取设备的设备节点路径(例如
/dev/sdb1
)。 - 参数:
udev_device
:设备对象。
- 返回值:成功时返回设备节点路径字符串,失败时返回
NULL
。 - 示例:
const char *devnode = udev_device_get_devnode(dev);
9. udev_device_get_devtype()
- 原型:
const char *udev_device_get_devtype(struct udev_device *udev_device);
- 作用:获取设备的类型(例如
"disk"
或"partition"
)。 - 参数:
udev_device
:设备对象。
- 返回值:成功时返回设备类型字符串,失败时返回
NULL
。 - 示例:
const char *devtype = udev_device_get_devtype(dev);
10. udev_device_unref()
- 原型:
void udev_device_unref(struct udev_device *udev_device);
- 作用:释放
udev_device
对象,减少其引用计数。 - 参数:
udev_device
:设备对象。
- 示例:
udev_device_unref(dev);
11. udev_monitor_unref()
- 原型:
void udev_monitor_unref(struct udev_monitor *udev_monitor);
- 作用:释放
udev_monitor
对象,减少其引用计数。 - 参数:
udev_monitor
:udev
监视器。
- 示例:
udev_monitor_unref(udev_monitor);
12. udev_unref()
- 原型:
void udev_unref(struct udev *udev);
- 作用:释放
udev
上下文对象,减少其引用计数。 - 参数:
udev
:udev
上下文。
- 示例:
udev_unref(udev);
相关文章:

Qt 监控USB设备的插入和移除
Qt 监控USB设备的插入和移除 flyfish Ubuntu22.04 Qt 6.2.4 CMakeLists.txt 内容 # 指定 CMake 的最低版本要求 cmake_minimum_required(VERSION 3.16)# 定义项目的名称和使用的编程语言 project(USBMonitor LANGUAGES CXX)# 开启自动 UIC,MOC 和 RCC 工具 set(…...

终于弄懂了Python自定义模块与代码复用
自定义模块与代码复用 在编写Python代码时,很多时候我们会遇到需要多次使用相同功能的情况。这时候,模块化编程就显得尤为重要。通过将常用的功能代码放入单独的模块中,我们可以轻松地进行代码复用,避免重复编写相同的代码&#…...

从无音响Windows 端到 有音响macOS 端实时音频传输播放
以下是从 Windows 端到 macOS 端传输音频的优化方案,基于上述链接中的思路进行调整: Windows 端操作 安装必要软件 安装 Python(确保版本兼容且已正确配置环境变量)。安装 PyAudio 库,可通过 pip install pyaudio 命令…...

直方图均衡化及Matlab实现
文章目录 直方图均衡化关键点及思路Matlab实现 直方图均衡化 直方图均衡化是一种图像增强技术,主要用于增强图像的对比度,特别是当图像的有用数据的对比度接近时效果显著。通过改变图像的直方图分布,直方图均衡化能够使图像的灰度值更加接近…...

设备接入到NVR管理平台EasyNVR多品牌NVR管理工具/设备的音视频配置参考
NVR管理平台EasyNVR是一款功能强大的安防视频监控平台,能够轻松实现视频流的导入、录像、存储和回放等功能。在将设备接入到海康NVR管理平台EasyNVR时,视音频配置是确保视频监控效果的重要步骤。本文将详细介绍如何将设备接入到EasyNVR平台,并…...

后端:Aop 面向切面编程
文章目录 1. Aop 初步学习面向切面编程,EnableAspectJAutoProxy2. AOP的核心概念3. 前置通知(Before)4. 后置通知(After)5. 返回通知(AfterReturning)6. 异常通知(AfterThrowing&…...

大数据机器学习算法与计算机视觉应用02:线性规划
Linear Programming Definition of linear programmingmax and min-cost max flowlinear program to solve minimax optimal strategies in gamesAlgoithms for linear programmingl1 regressionSeidel’s 2-dimensional linear programming algorithm linear program 线性规…...

godot——主题、Theme、StyleBox
我刚开始被这些术语吓到了,一直不敢去接触它们,都用的默认样式。现在好不容易有点思路了,记录下来。 下面看看怎么自定义样式。 1.先新建一个Theme 2.再次点击创建好的Theme 得到 图1 这样一个面板。(看不懂没事,继…...

深入理解接口测试:实用指南与最佳实践5.0(一)
✨博客主页: https://blog.csdn.net/m0_63815035?typeblog 💗《博客内容》:.NET、Java.测试开发、Python、Android、Go、Node、Android前端小程序等相关领域知识 📢博客专栏: https://blog.csdn.net/m0_63815035/cat…...

SQL面试题——飞猪SQL面试 重点用户
飞猪SQL面试题—重点用户 在一些场景中我们经常听到这样的一些描述,例如20%的用户贡献了80%的销售额,或者是20%的人拥有着80%的财富,你知道这样的数据是怎么算出来的吗 数据如下,uid 是用户的id ,amount是用户的消费金额 |uid|amount| ---…...

Angular 和 Vue2.0 对比
前言 :“业精于勤,荒于嬉;行成于思,毁于随” 很久没写博客了,大多记录少进一步探查。 Angular 和 Vue2.0 对比: 一.概念 1.1 Angular 框架: 是一款由谷歌开发的开源web前端框架(核…...

websocket服务器(协程风格)--swoole进阶篇
swoole的websocket服务器(协程风格)示例真不算友善,从头了解到尾,那还好,但是谁有那么多时间从头到尾了解。示例不够针对性,写websocket就该单独写websocket的东西,偏偏又加上http的东西。这里我来解读一下websocket服务器(协程风格)示例 <?php use Swoole\Http\…...

Windows C/C++ Socket 编程
承接上文:socket 编程 本文目录 Windows Client 端WSADATA 结构体WSAStartup() 函数SOCKET 以及 socket() 函数sockaddr_ininet_pton() 函数in_addr structmemcpy()connect() 函数send() 函数recv() 函数 Windows Server 端 在进行 socket 编程之前,你要…...

计算两个结构的乘法
在行列可自由变换的平面上,2点结构有3个 3点结构有6个 计算2*2 2a1*2a14a6 2a1*2a24a8 2a1*2a34a12 显然2a1*2a14a6因为这3个结构都分布在同一列上,就是整数乘法。2a1*2a2的结果有2种写法,一种外形像2a1细节为2a2,一种外形为2…...

学校服务器连接pycharm配置2
上一个可能还是有点问题,因为实际在跑的时候读取的其实是本地的anaconda,这个重新整了一下流程 首先在学校服务器先激活自己创建的虚拟环境,这里就不截图了 然后在pycharm里面打开设置 选择这个python解释器 这里有添加解释器 选择SSH …...

AI赋能电商:创新应用提升销售与用户体验
目录 一、引言 二、AI技术在电商领域的创新应用 三、AI技术提高电商销售效率和用户体验的实践路径 一、引言 随着人工智能(AI)技术的不断成熟,电商行业正迎来一场深刻的变革。AI技术在购物推荐、会员分类、商品定价等方面的创新应用&…...

详解kafka消息发送重试机制的案例
在 Kafka 生产者中实现消息发送的重试机制,可以通过配置 KafkaProducer 的相关属性来实现。以下是一些关键的配置项: retries:设置生产者发送失败后重试的次数。 retry.backoff.ms:设置生产者在重试前等待的时间。 buffer.memo…...

linux文本管理!!!
文章目录 第1章 文本过滤/查看命令1.echo:输出文本2.cat:合并文件或查看文件内容3.head:显示文件头部信息4.tail:显示文件尾部信息5.wc: 统计文本行号6.less:分页显示文件内容7.grep:文本过滤工具8.定向符号…...

软件设计师-计算机体系结构分类
计算机体系结构分类 Flynn分类法 根据不同的指令流数据流组织方式分类单指令流但数据流SISD,单处理器系统单指令多数据流SIMD,单指令流多数据流是一种采用一个控制器来控制多个处理器,同时对一组数据(又称“数据矢量”)中的每一…...

《基于深度学习的车辆行驶三维环境双目感知方法研究》
复原论文思路: 《基于深度学习的车辆行驶三维环境双目感知方法研究》 1、双目测距的原理 按照上述公式算的话,求d的话,只和xl-xr有关系,这样一来,是不是只要两张图像上一个测试点的像素位置确定,对应的深…...

jwt用户登录,网关给微服务传递用户信息,以及微服务间feign调用传递用户信息
1、引入jwt依赖 <dependency><groupId>io.jsonwebtoken</groupId><artifactId>jjwt</artifactId><version>0.9.1</version></dependency> 2、Jwt工具类,生成token以及解析token package com.niuniu.gateway.uti…...

ubontu安装anaconda
1.下载 Anaconda 安装脚本 2. 复制到服务器上/home/username文件夹中,进入文件夹,执行: bash Anaconda3-2024.10-1-Linux-x86_64.sh一直按回车,然后输入yes同意协议。 3. 初始化 Anaconda 环境,会自动配置环境变量&a…...

【Docker容器化技术】docker安装与配置、常用命令、容器数据卷、应用部署实战、Dockerfile、服务编排docker-compose、私有仓库
文章目录 一、Docker的安装与配置1、docker概述2、安装docker3、docker架构4、配置镜像加速器 二、Docker命令1、服务相关命令2、镜像相关命令3、容器相关命令 三、Docker容器数据卷1、数据卷概念及作用2、配置数据卷3、配置数据卷容器 四、Docker应用部署实战1、部署MySQL2、部…...

Python模拟A卷实操题
1.某机械公司生产两种产品。A的单件利润分别是100元,B的单件利润是150元。 每种产品由三种材料构成,现给出每种材料的库存(库存小于100000),求利润最大的生产方案。输入说明:第一行给出生产每件A产品所需要…...
Leetcode 检测相邻递增子数组
3349. 检测相邻递增子数组 I 给你一个由 n 个整数组成的数组 nums ,请你找出 k 的 最大值,使得存在 两个 相邻 且长度为 k 的 严格递增 子数组 。具体来说,需要检查是否存在从下标 a 和 b (a < b) 开始的 两个 子数组,并满…...

rockylinux 8安装 gcc11.2
方法 1:从源代码编译安装最新版本的 GCC 下载 GCC 源代码: 访问 GCC 官方网站下载最新版本的源代码,例如: wget https://ftp.gnu.org/gnu/gcc/gcc-11.2.0/gcc-11.2.0.tar.gz tar -xf gcc-11.2.0.tar.gz cd gcc-11.2.0安装依赖项&a…...

【蓝桥等考C++真题】蓝桥杯等级考试C++组第13级L13真题原题(含答案)-奇数序列排序
C L13 奇数序列排序 给定一个长度为N的正整数序列, 请将其中的所有奇数取出,并按增序(从小到大)输出。 输入: 共2行 第1行是一个正整数 N(不大于500); 第2行有 N 个正整数&#x…...

【AI】好用的AI记录
好用的AI 一、国内 KIMI通义 二、国外 GPT4Cursorv0...

linux安装boost.python
前言 boost.python库被用于C与Python代码间的交互,提供了两者间大部分数据类型的转换 相关环境 操作系统:Ubuntu 20.04 python版本:Python 3.8 boost版本:boost 1.78.0 安装 1.boost.python检查与卸载 在安装boost之前需要检…...

AI 扩展开发者思维方式:以 SQL 查询优化为例
在现代软件开发中,AI 技术的兴起让开发者的思维方式发生了显著变化。尤其是在 SQL 查询优化、代码重构以及算法设计等领域,AI 提供的建议不仅扩展了开发者的思考路径,还帮助他们发现以往没有意识到的潜在解决方案。 1. 传统思维模式下的 SQL…...