当前位置: 首页 > news >正文

系统讨论Qt的并发编程——逻辑上下文的分类

目录

前言

首先,讨论Qt里常见的三种上下文

同一线程的串行执行

同一线程的异步执行

多线程的执行

moveToThread办法


前言

笔者最近看了一个具备一定启发性质的Qt教程,在这里,笔者打算整理一下自己的笔记。分享在这里.

首先,讨论Qt里常见的三种上下文

Qt在逻辑层上,笔者认为可分成三种上下文:同线程的同步,同线程的异步和多线程。

一个典型的同线程之间的同步,说的是我们的处理在执行流上,是串行的。

main ---> A -----> B ------> C

比如说,A函数,B函数和C函数在上下文中,串行的执行。这样的上下文,非常的可靠,每一个指令流都串行的访问数据,不会出现数据竞争的问题,但是代价就是——非常的慢。笔者三年的编程经验告诉自己——这样的编程上下文,适用于流程在时序上非常严密的场景。

一个典型的同线程异步,事情就看起来很有趣了。实际上看起来像这样。

这样的技术,任何一个学习过OS的都知道,实际上是在应用层上对这个线程实现了分片,换而言之,我们在这个线程上实现了逻辑并行,物理串行的效果——看起来我们的代码在并发的执行,但是实际上是一个线程快无影的完成一大堆工作。所以,这样的编程范式属于高响应要求的场景,或者说,我们对性能谈不上太高的要求,只需要他们看起来在同时执行。

这个场景,就是真并发的场景了,不管怎么说,我们真的有多个线程执行多个代码流,我们的代码上下文被托付给了操作系统层级的线程完成我们的工作。我们老生长谈的数据竞争并发安全,就是在讨论这里的事情。

同一线程的串行执行

#include <QCoreApplication>
#include <QThread>
​
void workA() {qInfo() << "I am handling Work A" << "Thread:" << QThread::currentThread();QThread::currentThread()->sleep(1);qInfo() << "Work A Finished";
}
​
void workB() {qInfo() << "I am handling Work B" << "Thread:" << QThread::currentThread();QThread::currentThread()->sleep(1);qInfo() << "Work B Finished";
}
​
void workC() {qInfo() << "I am handling Work C" << "Thread:" << QThread::currentThread();QThread::currentThread()->sleep(1);qInfo() << "Work C Finished";
}
​
int main(int argc, char *argv[]) {QCoreApplication a(argc, argv);QThread::currentThread()->setObjectName("Thread Main");workA();workB();workC();return a.exec();
}

这个就是一个例子,我们的任务ABC按照串行依次执行代码。看我们的输出,就是这个主线程在依次的做任务ABC。

I am handling Work A Thread: QThread(0x1f0730e2860, name = "Thread Main")
Work A Finished
I am handling Work B Thread: QThread(0x1f0730e2860, name = "Thread Main")
Work B Finished
I am handling Work C Thread: QThread(0x1f0730e2860, name = "Thread Main")
Work C Finished

同一线程的异步执行

事实上,对于一些使用了QT类的场景,同一线程的异步执行仍然非常常见。一个经典的例子就是我们使用QTimer的singleshot办法触发一个函数的执行。你可以看到,我们实际上根本没有直接的调用,但是async_workA由于timer事件的通知作为槽执行了

#include <QCoreApplication>
#include <QThread>
#include <QTimer>
​
void async_workA() {qInfo() << "Lets's see the current thread: "<< QThread::currentThread()->objectName();QThread::currentThread()->sleep(1);qInfo() << "And A work done:)";
}
​
int main(int argc, char *argv[]) {QCoreApplication a(argc, argv);QThread::currentThread()->setObjectName("Main Thread");QTimer m;m.singleShot(1000, &async_workA);
​qInfo() << "Main Thread: " << QThread::currentThread()->objectName();qInfo() << "has reached its end!";return a.exec();
}
 

多线程的执行

现在我们开始上难度,那就是多线程的执行。

TestObject.cpp

#include "TestObject.h"
​
#include <QDebug>
#include <QThread>
#include <QTimer>
TestObject::TestObject(QObject *parent) : QObject{parent} {}
​
void TestObject::runMe() {qInfo() << QThread::currentThread()->objectName() << " Running";QTimer::singleShot(1000, this, &TestObject::internel_shot);qInfo() << QThread::currentThread()->objectName() << " Finish";
}
​
void TestObject::internel_shot() { qInfo() << "Shot"; }

Main.cpp

#include <QCoreApplication>
#include <QThread>
​
#include "TestObject.h"
int main(int argc, char *argv[]) {QCoreApplication a(argc, argv);
​QThread::currentThread()->setObjectName("Main Thread");
​qInfo() << "Ready to shot a thread";
​QThread thread;thread.setObjectName("Thread Worker");
​TestObject test;
​test.moveToThread(&thread);
​QObject::connect(&thread, &QThread::started, &test, &TestObject::runMe);
​thread.start();
​qInfo() << "Finish " << QThread::currentThread()->objectName();
​return a.exec();
}

这里有一个新的方法,简单说一下:

moveToThread办法

moveToThread 是 Qt 中用于将对象移到指定线程的函数。必须说明的是——我们的QT信号与槽机制是不跨线程的,也就是说,我们没办法使用信号与槽机制链接在A线程的a对象的信号与在B线程的b对象的槽。对于非 GUI 类的对象,可以通过调用 moveToThread 将它们从主线程或其他线程转移到目标线程。这样,目标线程就会负责该对象的事件处理和信号槽连接。

这样,我们就会将我们的TestObject放置到我们的新线程中,换而言之,任何一个将会在B线程工作但是创建于A线程的物体,请使用这个方法。

相关文章:

系统讨论Qt的并发编程——逻辑上下文的分类

目录 前言 首先&#xff0c;讨论Qt里常见的三种上下文 同一线程的串行执行 同一线程的异步执行 多线程的执行 moveToThread办法 前言 笔者最近看了一个具备一定启发性质的Qt教程&#xff0c;在这里&#xff0c;笔者打算整理一下自己的笔记。分享在这里. 首先&#xff0c…...

《Linux Shell 脚本深度探索:原理与高效编程》

1. 基本结构 Shebang 行 #!/bin/bash # Shebang 行指定了脚本使用的解释器。 /bin/bash 表示使用 Bash 解释器执行脚本。 注释 # 这是注释&#xff0c;不会被执行 2. 变量 定义变量 variable_namevalue # 不需要加 $ 来定义变量。 # 变量名不能包含空格或特殊字符。 访…...

深入剖析:基于红黑树实现自定义 map 和 set 容器

&#x1f31f; 快来参与讨论&#x1f4ac;&#xff0c;点赞&#x1f44d;、收藏⭐、分享&#x1f4e4;&#xff0c;共创活力社区。&#x1f31f; 在 C 标准模板库&#xff08;STL&#xff09;的大家庭里&#xff0c;map和set可是超级重要的关联容器成员呢&#x1f60e;&#x…...

在大数据项目中如何设计和优化数据模型

在大数据项目中&#xff0c;设计和优化数据模型是一个涉及多个步骤和维度的复杂过程。以下是我通常采取的方法&#xff1a; 一、数据模型设计 明确业务需求&#xff1a; 深入了解项目的业务场景和目标&#xff0c;明确数据模型需要解决的具体问题。与业务团队紧密合作&#xf…...

JavaScript querySelector()、querySelectorAll() CSS选择器解析(DOM元素选择)

文章目录 基于querySelector系列方法的CSS选择器深度解析一、方法概述二、基础选择器类型1. 类型选择器2. ID选择器3. 类选择器4. 属性选择器 三、组合选择器1. 后代组合器2. 子元素组合器3. 相邻兄弟组合器4. 通用兄弟组合器 四、伪类与伪元素1. 结构伪类2. 状态伪类3. 内容伪…...

Linux系统中处理子进程的终止问题

1. 理解子进程终止的机制 在Unix/Linux系统中&#xff0c;当子进程终止时&#xff0c;会向父进程发送一个SIGCHLD信号。父进程需要捕捉这个信号&#xff0c;并通过调用wait()或waitpid()等函数来回收子进程的资源。这一过程被称为“回收僵尸进程”。 如果父进程没有及时调用w…...

Docker 不再难懂:快速掌握容器命令与架构原理

1. Docker 是容器技术的一种 容器&#xff08;Container&#xff09;概述 容器&#xff08;Container&#xff09;是一种轻量级的虚拟化技术&#xff0c;它将应用程序及其所有依赖环境打包在一个独立的、可移植的运行时环境中。容器通过操作系统级的虚拟化提供隔离&#xff0…...

取消票证会把指定的票证从数据库中删除,同时也会把票证和航班 等相关表中的关联关系一起删除。但在删除之前,它会先检查当前用户是否拥有这张票

在做航班智能客服问答系统时会遇到取消票证的场景&#xff0c;这里涉及数据库的操作时会把指定的票证从数据库中删除&#xff0c;同时也会把票证和航班等相关表中的关联关系一起删除。但在删除之前&#xff0c;需要先检查当前用户是否拥有这张票&#xff0c;只有票主才有权限取…...

力扣-贪心-763 划分字母区间

思路 先统计字符串中每一个字母出现的最后下标&#xff0c;然后从end初始化为第一个字母出现的最后下标&#xff0c;在i<end时&#xff0c;不断更新end&#xff0c;因为一旦囊括新的字母就最起码要遍历到新字母出现的最后下标&#xff0c;在i>end时&#xff0c;说明遍历…...

【Redis 原理】网络模型

文章目录 用户空间 && 内核空间阻塞IO非阻塞IO信号驱动IO异步IOIO多路复用selectpollepoll Web服务流程Redis 网络模型Redis单线程网络模型的整个流程Redis多线程网络模型的整个流程 用户空间 && 内核空间 为了避免用户应用导致冲突甚至内核崩溃&#xff0c;用…...

cpp中的继承

一、继承概念 在cpp中&#xff0c;封装、继承、多态是面向对象的三大特性。这里的继承就是允许已经存在的类&#xff08;也就是基类&#xff09;的基础上创建新类&#xff08;派生类或者子类&#xff09;&#xff0c;从而实现代码的复用。 如上图所示&#xff0c;Person是基类&…...

DeepSeek全栈接入指南:从零到生产环境的深度实践

第一章:DeepSeek技术体系全景解析 1.1 认知DeepSeek技术生态 DeepSeek作为新一代人工智能技术平台,构建了覆盖算法开发、模型训练、服务部署的全链路技术栈。其核心能力体现在: 1.1.1 多模态智能引擎 自然语言处理:支持文本生成(NLG)、语义理解(NLU)、情感分析等计算…...

CSS 真的会阻塞文档解析吗?

在网页开发领域&#xff0c;一个常见的疑问是 CSS 是否会阻塞文档解析。理解这一问题对于优化网页性能、提升用户体验至关重要。要深入解答这个问题&#xff0c;需要从浏览器渲染网页的原理说起。 浏览器渲染网页的基本流程 浏览器在接收到 HTML 文档后&#xff0c;会依次进行…...

大模型的UI自动化:Cline 使用Playwright MCP Server完成测试

大模型的UI自动化:Cline 使用Playwright MCP Server完成测试 MCP MCP(Model Context Protocol),是一个开发的协议,标准化了应用程序如何为大模型提供上下文。MCP提供了一个标准的为LLM提供数据、工具的方式,使用MCP会更容易的构建Agent或者是基于LLM的复杂工作流。 最近…...

碰撞检测 | 图解凸多边形分离轴定理(附ROS C++可视化)

目录 0 专栏介绍1 凸多边形碰撞检测2 多边形判凸算法3 分离轴定理(SAT)4 算法仿真与可视化4.1 核心算法4.2 仿真实验 0 专栏介绍 &#x1f525;课设、毕设、创新竞赛必备&#xff01;&#x1f525;本专栏涉及更高阶的运动规划算法轨迹优化实战&#xff0c;包括&#xff1a;曲线…...

Python 基本数据类型

目录 1. 字符串&#xff08;String&#xff09; 2. 列表&#xff08;List&#xff09; 3. 字典&#xff08;Dictionary&#xff09; 4. 集合&#xff08;Set&#xff09; 5. 数字&#xff08;Number&#xff09; 6. 布尔值&#xff08;Boolean&#xff09; 1. 字符串&…...

突破“第一崇拜“:五维心理重构之路

一、视频介绍 在这个崇尚"第一"的时代&#xff0c;我们如何找到自己的独特价值&#xff1f;本视频将带您踏上五维心理重构之旅&#xff0c;从诗意人生的角度探讨如何突破"圣人之下皆蝼蚁"的局限。我们将穿越人生的不同阶段&#xff0c;从青春的意气风发到…...

KubeKey一键安装部署k8s集群和KubeSphere详细教程

目录 一、KubeKey简介 二、k8s集群KubeSphere安装 集群规划 硬件要求 Kubernetes支持版本 操作系统要求 SSH免密登录 配置集群时钟 所有节点安装依赖 安装docker DNS要求 存储要求 下载 KubeKey 验证KubeKey 配置集群文件 安装集群 验证命令 登录页面 一、Ku…...

UE5网络通信架构解析

文章目录 前言一、客户端-服务器架构&#xff08;C/S Model&#xff09;二、对等网络架构&#xff08;P2P&#xff0c;非原生支持&#xff09;三、混合架构&#xff08;自定义扩展&#xff09;四、UE5网络核心机制 前言 UE5的网络通信主要基于客户端-服务器&#xff08;C/S&am…...

实验3 知识表示与推理

实验3 知识表示与推理 一、实验目的 &#xff08;1&#xff09;掌握知识和知识表示的基本概念&#xff0c;理解其在AI中的深刻含义与意义&#xff1b; &#xff08;2&#xff09;熟悉AI中常用的知识表示方法的优缺点及其应用场景&#xff1b; &#xff08;3&#xff09;掌握产…...

万象视界灵坛部署案例:智能硬件产品图‘工业设计感’‘科技感’评分系统

万象视界灵坛部署案例&#xff1a;智能硬件产品图工业设计感科技感评分系统 1. 项目背景与价值 在智能硬件产品开发过程中&#xff0c;产品外观设计的"工业设计感"和"科技感"是影响消费者购买决策的重要因素。传统评估方式依赖人工评审&#xff0c;存在主…...

简历怎么写:我做了什么,取得了什么成果,凸显JD 关键词

简历怎么写:我做了什么,取得了什么成果;凸显JD 关键词 这里写目录标题 简历怎么写:我做了什么,取得了什么成果;凸显JD 关键词 一、先搞清楚:HR到底是做什么的? 二、HR是怎么筛选简历的?说出来你可能不信 1. 关键词!关键词!关键词! 2. 这些信息,写了反而减分 3. 一…...

【倒计时87天】2026奇点大会唯一闭门议程曝光:AGI记忆持久化标准草案(ISO/IEC JTC 1/SC 42 WG12内部修订版首次流出)

第一章&#xff1a;2026奇点智能技术大会&#xff1a;AGI与记忆系统 2026奇点智能技术大会(https://ml-summit.org) 本届大会首次将“记忆系统”列为AGI架构的核心支柱&#xff0c;而非传统意义上的辅助模块。来自DeepMind、Tsinghua AGI Lab与Neuromorphic Memory Consortiu…...

别再为SURF/SIFT发愁了!Ubuntu 20.04下OpenCV_contrib离线安装全攻略(含预编译模型包)

Ubuntu 20.04下OpenCV_contrib离线安装终极指南&#xff1a;预编译模型包与避坑手册 在计算机视觉开发中&#xff0c;SURF、SIFT等经典特征提取算法依然是许多项目的基石。然而&#xff0c;当你在Ubuntu 20.04上尝试安装OpenCV_contrib扩展库时&#xff0c;可能会遇到各种网络下…...

ssm社区物业信息管理系统小程序(文档+源码)_kaic

系统实现登录模块的实现系统的登录窗口是用户的入口&#xff0c;用户只有在登录成功后才可以进入访问。通过在登录提交表单&#xff0c;后台处理判断是否为合法用户&#xff0c;进行页面跳转&#xff0c;进入系统中去。登录合法性判断过程&#xff1a;用户输入账号和密码后&…...

新手避坑指南:用Arduino UNO和NRF24L01模块做无线通信,这5个细节不注意就白忙活

Arduino UNO与NRF24L01无线通信实战&#xff1a;新手必知的5个关键细节 第一次接触无线通信项目时&#xff0c;我像大多数创客爱好者一样选择了NRF24L01模块。这个价格亲民的2.4GHz无线收发器看似简单&#xff0c;却在初次使用时让我经历了整整三天的调试噩梦——两个模块死活无…...

AGI在员工体验管理中的隐秘应用:从情绪语义分析到个性化发展路径生成(仅限头部科技公司内部验证)

第一章&#xff1a;AGI在员工体验管理中的隐秘应用&#xff1a;从情绪语义分析到个性化发展路径生成&#xff08;仅限头部科技公司内部验证&#xff09; 2026奇点智能技术大会(https://ml-summit.org) 在硅谷与西雅图的三座超算中心内&#xff0c;某头部AI原生企业已将AGI模型…...

【OpenCV 实战指南】特征匹配:从暴力匹配到实战调优

1. 暴力匹配基础&#xff1a;从理论到OpenCV实现 第一次接触特征匹配时&#xff0c;我被这个看似简单实则精妙的技术深深吸引了。想象一下&#xff0c;你手上有两张不同角度拍摄的同一栋建筑的照片&#xff0c;如何让计算机自动找到两张照片中相同的窗户或装饰&#xff1f;这就…...

【实战指南】FreeRTOS 10.4.6源码解析与STM32F429移植全流程

1. FreeRTOS 10.4.6源码获取与解析 第一次接触FreeRTOS源码时&#xff0c;我对着官网密密麻麻的目录树发懵——这堆文件到底哪些才是核心&#xff1f;后来踩过几次坑才明白&#xff0c;Source和portable这两个文件夹就是整个系统的灵魂所在。以STM32F429为例&#xff0c;我们从…...

今天不看SITS2026这页PPT,明年招标书里将彻底消失“传统机器人”术语

第一章&#xff1a;SITS2026演讲&#xff1a;AGI与机器人结合 2026奇点智能技术大会(https://ml-summit.org) 核心范式转变 传统机器人系统依赖预编程行为树与模块化感知-决策-执行链路&#xff0c;而SITS2026展示的AGI驱动架构将大语言模型&#xff08;LLM&#xff09;与具身…...