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

从pthread到std::jthread:一个C++老鸟的多线程编程进化史

从pthread到std::jthread一个C老鸟的多线程编程进化史记得第一次接触多线程编程是在2008年那时我刚从学校毕业加入了一家做网络设备的公司。我们的产品需要处理大量并发连接而当时的C标准库还没有原生线程支持。于是我开始了与pthread的漫长斗争。1. 石器时代与pthread共舞的日子在C11之前Linux下的多线程编程几乎等同于使用pthread库。那些年每个C程序员都要掌握一套特殊的生存技能#include pthread.h void* thread_func(void* arg) { int* num (int*)arg; printf(Thread received: %d\n, *num); return NULL; } int main() { pthread_t tid; int arg 42; pthread_create(tid, NULL, thread_func, arg); pthread_join(tid, NULL); return 0; }这段简单的代码背后隐藏着无数坑点类型安全缺失必须使用void*进行参数传递完全失去了C的类型检查优势资源管理困难忘记调用pthread_join会导致内存泄漏错误处理繁琐每个pthread函数调用后都要检查返回值跨平台差异不同Unix-like系统的实现细节有微妙差别最让我记忆深刻的是2010年的一次生产事故。当时我们有一个服务因为忘记pthread_join导致线程资源泄漏运行两周后就耗尽了系统资源。那次事故让我写了整整三天的核心转储分析报告。2. 工业革命C11带来的std::thread曙光当C11标准发布时std::thread就像黑暗中的一束光。对比之前的pthread它带来了几个革命性的改进特性pthreadstd::thread类型安全无使用void*有模板参数资源管理手动基于RAII错误处理返回错误码抛出异常可移植性仅限于POSIX系统跨平台函数对象支持仅支持C函数支持lambda和函数对象第一次用std::thread重构旧代码时的快感至今难忘#include thread #include iostream void worker(int value) { std::cout Thread got: value std::endl; } int main() { int value 42; std::thread t(worker, value); t.join(); return 0; }但std::thread并非完美。在大型项目中我们很快发现了新的挑战线程生命周期管理如果忘记join或detach程序会terminate异常安全线程函数抛出异常时处理起来很棘手线程取消没有标准化的线程停止机制2015年我在一个高并发交易系统中就踩到了大坑。当时我们需要动态调整工作线程数量但发现优雅停止线程几乎是不可能的任务。最终我们不得不实现了一套基于条件变量的复杂停止机制。3. 智能时代C20的std::jthread革命C20引入的std::jthread解决了std::thread最痛的两个问题自动资源管理析构时自动join再也不用担心忘记join导致的问题内置停止机制通过stop_token实现线程间协作式停止3.1 stop_token的魔法stop_token机制的核心思想是协作式取消而非强制终止。看看这个典型的生产者-消费者模式实现#include thread #include iostream #include queue #include mutex #include condition_variable void producer(std::stop_token stop, std::queueint q, std::mutex mtx, std::condition_variable cv) { for(int i 0; !stop.stop_requested(); i) { { std::lock_guard lock(mtx); q.push(i); } cv.notify_one(); std::this_thread::sleep_for(std::chrono::milliseconds(100)); } } void consumer(std::stop_token stop, std::queueint q, std::mutex mtx, std::condition_variable cv) { while(!stop.stop_requested()) { std::unique_lock lock(mtx); cv.wait(lock, stop, [q]{ return !q.empty(); }); if(stop.stop_requested()) break; int val q.front(); q.pop(); lock.unlock(); std::cout Consumed: val std::endl; } } int main() { std::queueint q; std::mutex mtx; std::condition_variable cv; std::jthread prod(producer, std::ref(q), std::ref(mtx), std::ref(cv)); std::jthread cons(consumer, std::ref(q), std::ref(mtx), std::ref(cv)); std::this_thread::sleep_for(std::chrono::seconds(1)); // 无需手动join析构时会自动处理 return 0; }这个例子展示了std::jthread的几个关键优势自动资源清理即使发生异常jthread也会确保线程正确join优雅停止通过stop_token检查停止请求线程可以安全清理资源条件变量集成condition_variable支持stop_token感知的等待3.2 性能与取舍std::jthread不是免费的午餐它带来的便利性是有代价的每个jthread对象需要额外存储stop_statestop_token检查会引入轻微运行时开销比std::thread占用更多内存通常多16-32字节在极端性能敏感的场景你可能还需要回归到std::thread甚至pthread。但在90%的应用中jthread的开销完全可以接受。4. 实战现代化线程池设计结合这些年积累的经验我总结出了一个现代化的线程池实现模式class ThreadPool { public: explicit ThreadPool(size_t num_threads) { for(size_t i 0; i num_threads; i) { workers_.emplace_back([this](std::stop_token st) { while(!st.stop_requested()) { std::functionvoid() task; { std::unique_lock lock(queue_mutex_); queue_cv_.wait(lock, st, [this]{ return !tasks_.empty(); }); if(st.stop_requested()) return; task std::move(tasks_.front()); tasks_.pop(); } task(); } }); } } ~ThreadPool() { for(auto worker : workers_) { worker.request_stop(); } queue_cv_.notify_all(); } templatetypename F void enqueue(F f) { { std::lock_guard lock(queue_mutex_); tasks_.emplace(std::forwardF(f)); } queue_cv_.notify_one(); } private: std::vectorstd::jthread workers_; std::queuestd::functionvoid() tasks_; std::mutex queue_mutex_; std::condition_variable_any queue_cv_; };这个实现有几个值得注意的特点异常安全使用jthread确保线程总会正确join优雅关闭析构时请求所有线程停止停止感知条件变量配合stop_token工作通用任务使用std::function支持任意可调用对象在最近的一个数据处理项目中这个线程池设计帮助我们减少了约40%的线程管理代码同时显著提高了系统的可靠性。

相关文章:

从pthread到std::jthread:一个C++老鸟的多线程编程进化史

从pthread到std::jthread:一个C老鸟的多线程编程进化史 记得第一次接触多线程编程是在2008年,那时我刚从学校毕业,加入了一家做网络设备的公司。我们的产品需要处理大量并发连接,而当时的C标准库还没有原生线程支持。于是&#xf…...

WebRTC降噪模块ns_core源码‘导游’:从频谱平坦度到维纳滤波,一次搞懂噪声估计

WebRTC降噪模块ns_core源码深度解析:从频谱分析到实时噪声抑制 在实时音视频通信领域,背景噪声一直是影响通话质量的关键因素。WebRTC作为开源实时通信方案的标杆,其噪声抑制模块(ns_core)通过多维度信号分析实现了出色的降噪效果。本文将带您…...

别再被参数忽悠了!5分钟搞懂手机电脑屏幕怎么选:从LCD、OLED到IPS、TN的实用避坑指南

别再被参数忽悠了!5分钟搞懂手机电脑屏幕怎么选:从LCD、OLED到IPS、TN的实用避坑指南 走进数码卖场,面对琳琅满目的手机、笔记本和显示器,你是否曾被商家宣传的"100% sRGB色域"、"144Hz刷新率"、"IPS硬屏…...

猫抓浏览器插件:免费开源的网页资源嗅探终极指南

猫抓浏览器插件:免费开源的网页资源嗅探终极指南 【免费下载链接】cat-catch 猫抓 浏览器资源嗅探扩展 / cat-catch Browser Resource Sniffing Extension 项目地址: https://gitcode.com/GitHub_Trending/ca/cat-catch 猫抓(Cat-Catch&#xff0…...

KeymouseGo深度解析:如何通过Python实现跨平台鼠标键盘自动化

KeymouseGo深度解析:如何通过Python实现跨平台鼠标键盘自动化 【免费下载链接】KeymouseGo 类似按键精灵的鼠标键盘录制和自动化操作 模拟点击和键入 | automate mouse clicks and keyboard input 项目地址: https://gitcode.com/gh_mirrors/ke/KeymouseGo K…...

PID参数自整定实战解析:从继电反馈到智能调节的完整实现

1. 继电反馈自整定的核心原理 第一次接触PID参数自整定时,我和大多数工程师一样被各种数学公式绕得头晕。直到在某个深夜调试恒温箱时,偶然发现继电器反复开关会引发温度震荡,才突然理解继电反馈法的精妙——它本质上是用极限环振荡这个物理现…...

C++ vector容器的用法

一、头文件#include<iostream>#include<vector>using namespace std;或者万能头文件 #include<bits/stdc.h>using namespace std;二、定义vector1、定义空的vector&#xff0c;后续存元素&#xff1a;vector<int> v1;2、指定大小&#xff0c;初始值…...

构建企业级金融数据集成架构:基于YahooFinanceApi的高性能解决方案

构建企业级金融数据集成架构&#xff1a;基于YahooFinanceApi的高性能解决方案 【免费下载链接】YahooFinanceApi A handy Yahoo! Finance api wrapper, based on .NET Standard 2.0 项目地址: https://gitcode.com/gh_mirrors/ya/YahooFinanceApi 在当今金融科技快速发…...

从源码到镜像:手把手教你定制并容器化FastGPT开发环境

1. 为什么需要定制FastGPT开发环境&#xff1f; FastGPT作为一款开源的AI应用框架&#xff0c;官方提供的标准镜像虽然开箱即用&#xff0c;但实际开发中总会遇到个性化需求。比如上周我帮一家教育机构部署时&#xff0c;对方要求替换所有品牌标识、调整界面配色&#xff0c;甚…...

速腾R128雷达+海康全景相机ROS数据采集避坑实录:从‘雷达不转’到MATLAB标定的完整流程

速腾R128雷达与海康全景相机ROS数据采集实战&#xff1a;从硬件配置到MATLAB标定的全流程解析 实验室里那台速腾R128激光雷达和海康威视全景相机的组合&#xff0c;成了我这学期最难忘的"搭档"。第一次尝试多传感器数据采集时&#xff0c;我经历了从兴奋到崩溃再到豁…...

Qwen3-TTS-VoiceDesign多语种落地:跨境电商商品解说(日/韩/德/法语)语音批量生成方案

Qwen3-TTS-VoiceDesign多语种落地&#xff1a;跨境电商商品解说&#xff08;日/韩/德/法语&#xff09;语音批量生成方案 1. 项目背景与价值 跨境电商卖家经常面临一个共同难题&#xff1a;如何为不同国家的客户提供本地化的商品解说语音&#xff1f;传统方案需要雇佣多语种配…...

新手避坑必看!OpenClaw v2.6.2 Windows 本地AI智能体部署教程,报错速解不踩雷

2026年开源本地AI智能体持续升温&#xff0c;OpenClaw&#xff08;昵称小龙虾&#xff09;凭借“零代码操作、本地安全运行、全场景自动化”的核心优势&#xff0c;成为办公族和技术新手的必备工具&#xff0c;GitHub星标突破28万&#xff0c;彻底解决了传统AI工具“技术门槛高…...

如何快速掌握游戏化编程学习:CodeCombat新手完整入门指南

如何快速掌握游戏化编程学习&#xff1a;CodeCombat新手完整入门指南 【免费下载链接】codecombat Game for learning how to code. 项目地址: https://gitcode.com/gh_mirrors/co/codecombat CodeCombat是一款创新的游戏化编程学习平台&#xff0c;将编程教育与角色扮演…...

HFSS仿真报错别慌!从‘Port supports an additional mode’到‘Poor convergence’的保姆级排查手册

HFSS仿真报错排查实战指南&#xff1a;从端口模式到收敛问题的深度解析 第一次打开HFSS仿真结果看到满屏英文报错时&#xff0c;那种手足无措的感觉我至今记忆犹新。作为一款功能强大的电磁场仿真软件&#xff0c;HFSS在带来高精度计算结果的同时&#xff0c;也因其复杂的设置和…...

5分钟搞定Docker私有仓库:Nexus3最新版搭建与镜像上传全流程

5分钟搞定Docker私有仓库&#xff1a;Nexus3最新版搭建与镜像上传全流程 在当今云原生技术快速发展的背景下&#xff0c;企业级容器镜像管理已成为DevOps流程中不可或缺的一环。对于中小团队而言&#xff0c;直接使用公共镜像仓库既存在安全风险&#xff0c;又难以满足定制化需…...

不只是跑通!用YOLOv8+RTX 3060做视频分析,从参数调优到性能监控的保姆级实践

YOLOv8与RTX 3060的工业级视频分析实战&#xff1a;从参数调优到系统监控 当你的YOLOv8模型能在RTX 3060上跑通基础推理时&#xff0c;真正的挑战才刚刚开始。在安防监控、工业质检等实际场景中&#xff0c;我们需要的不只是"能运行"&#xff0c;而是稳定、高效、可监…...

你的智能硬件还只能‘哔哔’响?试试用ESP32和minimp3做个网络电台或语音提示器

用ESP32和minimp3打造智能硬件的语音交互系统 想象一下&#xff0c;当你走进房间时&#xff0c;温湿度传感器不再只是默默记录数据&#xff0c;而是用自然语音提醒你"当前室内温度26℃&#xff0c;湿度45%&#xff0c;适宜居住"&#xff1b;或者你的智能花园系统在土…...

一键生成动漫头像!梦幻动漫魔法工坊实战应用分享

一键生成动漫头像&#xff01;梦幻动漫魔法工坊实战应用分享 1. 为什么你需要这个动漫头像生成器 在社交媒体时代&#xff0c;一个独特的动漫头像能让你在朋友圈、游戏社区或社交平台中脱颖而出。传统方式要么需要绘画技能&#xff0c;要么需要付费定制&#xff0c;而"梦…...

在VMware Workstation 17上部署openEuler 24.03 LTS SP1:从安装到静态IP配置全流程

1. 环境准备与镜像获取 最近在帮团队搭建国产操作系统开发环境时&#xff0c;发现openEuler 24.03 LTS SP1这个版本特别适合作为基础平台。相比其他Linux发行版&#xff0c;它在安全性和稳定性方面表现突出&#xff0c;特别适合需要长期维护的项目。下面我就把在VMware Worksta…...

景区BGM还在用《成都》和班得瑞?2026景区公播音乐避坑与经营升维指南

告别全园同响 侵权风险&#xff0c;用声音撬动游客停留时长与二次消费【本文能帮你解决】 ✅ 景区公播音乐同质化严重&#xff0c;游客没记忆点怎么办 ✅ 全园乱播音乐&#xff0c;破坏沉浸式体验如何整改 ✅ 背景音乐侵权风险高&#xff0c;低成本合规方案 ✅ 智慧景区广播系…...

用AI搞定STK12.2 Python二次开发:实测ChatGPT辅助写卫星仿真代码

用AI加速STK12.2卫星仿真开发&#xff1a;Python与ChatGPT高效协作指南 当面对STK12.2庞大的COM接口和复杂的对象模型时&#xff0c;即使是经验丰富的工程师也常感到无从下手。传统学习路径需要耗费大量时间研读文档和接口关系图&#xff0c;而如今&#xff0c;借助大语言模型的…...

GD32F303实战 ----- 定时器PWM驱动LED实现渐变调光

1. 从零开始理解PWM调光 想象一下老式台灯的旋钮开关&#xff0c;旋转角度越大灯光越亮——这种通过调节"通电时间比例"来控制亮度的原理&#xff0c;就是PWM&#xff08;脉冲宽度调制&#xff09;技术的雏形。在GD32F303开发板上&#xff0c;我们通过定时器产生精确…...

保姆级教程:让Ollama的Embedding API完美兼容OpenAI客户端(Python/Node.js双版本)

无缝对接Ollama与OpenAI Embedding API的跨语言实践指南 当开发者需要在本地环境中运行大语言模型时&#xff0c;Ollama已成为许多人的首选工具。然而&#xff0c;现有项目中大量基于OpenAI官方SDK编写的代码如何平滑迁移到Ollama服务上&#xff0c;成为了一个现实的技术挑战。…...

Zynq-7000和Ultrascale EMIO引脚分配差异对比:避免跨平台开发的常见错误

Zynq-7000与Ultrascale EMIO引脚分配差异全解析&#xff1a;从原理到避坑实战 在FPGA与SoC混合架构设计中&#xff0c;Xilinx Zynq系列因其灵活的可编程逻辑与强大处理能力而广受欢迎。但当开发者需要在Zynq-7000和Ultrascale平台间迁移项目时&#xff0c;EMIO引脚编号的差异往…...

从‘特斯拉线圈’到‘家庭插座’:聊聊交流电系统中‘地线’的前世今生与关键作用

从‘特斯拉线圈’到‘家庭插座’&#xff1a;交流电系统中地线的进化史与安全逻辑 1884年&#xff0c;当尼古拉特斯拉带着他的交流电系统设计图纸踏上美国土地时&#xff0c;这位塞尔维亚发明家或许不会想到&#xff0c;他手中那些缠绕着铜线的线圈&#xff0c;将彻底改变人类使…...

MATLAB与STK互联实战:自动化构建Walker星座的完整指南

1. MATLAB与STK互联基础&#xff1a;为什么选择自动化构建Walker星座 第一次接触卫星星座仿真时&#xff0c;我也像大多数人一样在STK界面里手动点击操作。直到某次需要调整30多次参数反复测试&#xff0c;连续加班到凌晨三点后&#xff0c;我才意识到必须找到更高效的方法。MA…...

K-Means聚类算法完整指南:从原理到实战

Python K-means聚类算法完整实战&#xff1a;用户分群详细代码注释聚类是数据分析中最常用的无监督学习方法&#xff0c;而K-means是最经典、最广泛使用的聚类算法。本文用一个真实业务场景——电商用户分群&#xff0c;从零带你掌握K-means的完整实战流程&#xff0c;每行代码…...

从 Spotlight 到 Raycast:一个 Mac 用户的效率工具进化史

1. 从Spotlight到Raycast&#xff1a;我的效率工具进化之路 第一次接触Mac电脑时&#xff0c;系统自带的Spotlight搜索让我眼前一亮。按下Command空格键&#xff0c;输入几个字母就能快速启动应用或查找文件&#xff0c;这在Windows系统上是完全不同的体验。但随着使用时间增长…...

Python 正则表达式:高级技巧与性能优化

Python 正则表达式&#xff1a;高级技巧与性能优化 核心概念与原理 正则表达式是一种用于匹配字符串中字符组合的模式&#xff0c;在Python中通过re模块实现&#xff0c;是处理文本数据的强大工具。 正则表达式的基本组成 组件描述示例字符匹配自身a 匹配 "a"元字符具…...

基于wxauto与Coze API,打造专属微信群AI助手

1. 为什么需要微信群AI助手&#xff1f; 最近两年AI技术发展迅猛&#xff0c;各种大模型层出不穷。但很多朋友发现&#xff0c;虽然AI很强大&#xff0c;但真正用起来却不太方便。比如想要在微信群里使用AI&#xff0c;要么得手动复制粘贴问题&#xff0c;要么得频繁切换应用&a…...