【C++入门到精通】C++ thread线程库 [ C++入门 ]

在这里插入图片描述

阅读导航

  • 引言
  • 一、thread类的简单介绍
  • 二、thread类的用法
    • 1. 创建线程
    • 2. 使用 Lambda 表达式
    • 3. 传递参数给线程
    • 4. 线程的 join 和 detach
    • 5. 检查线程是否可 join
    • 6. 线程的 ID
    • 7. 线程的移动语义
    • 8. 线程的析构
    • 🚨 注意事项
  • 三、线程函数参数
  • 温馨提示

引言

C++ thread线程库是C++11标准引入的一个强大工具,它提供了一种便捷的方式来创建和管理线程,使得并行编程变得更加容易和高效。这个库支持线程的创建、同步、互斥以及线程局部存储等功能。通过使用std::thread类,开发者可以轻松地创建新的线程来执行任务,并使用join()方法来等待线程完成。此外,线程库还包括了std::mutexstd::lock_guard等同步原语,以帮助管理线程间的资源访问,防止数据竞争和死锁。线程局部存储std::thread::idthread_local关键字则允许线程拥有自己的局部数据,这在多线程环境中非常有用。总的来说,C++ thread线程库为C++开发者提供了一个功能全面、易于使用的多线程编程解决方案。让我们一起开始这段关于thread线程库的学习之旅吧。

一、thread类的简单介绍

std::thread 类是C++11标准库中的一个核心组件,用于创建和管理独立的线程。它允许开发者通过传递一个函数或可调用对象来初始化线程,执行并行任务。线程对象的生命周期控制着线程的执行,而通过join()detach()方法,可以控制线程的同步和分离。此外,std::thread还提供了线程ID和状态检查功能,帮助开发者进行线程管理和异常处理,确保程序的稳定性和效率。
🚨注意:要使用线程库中的线程,必须包含<thread>头文件。线程类官方介绍文档

下面这个表格包含了 std::thread 类的构造函数、赋值运算符、比较运算符以及一些用于线程管理的成员函数。这些函数提供了创建、管理、比较和销毁线程的能力.

函数名功能描述
id get_id()返回线程的唯一标识符。
bool joinable()检查线程是否可 join,即是否还在运行。
void join()等待线程结束执行。
void detach()将线程与 std::thread 对象分离,使其在后台独立运行。
void swap(std::thread& other)与另一个 std::thread 对象交换线程。
thread::native_handle_type native_handle()返回线程的原生句柄,用于操作系统特定的线程操作。
bool operator==(const thread& other) const比较两个线程是否相同。
bool operator!=(const thread& other) const比较两个线程是否不同。
thread() noexcept默认构造函数,创建一个未关联线程的 std::thread 对象。
thread(nullptr_t) noexcept构造一个未关联线程的 std::thread 对象。
explicit thread(Callable&& func, Args&&… args)构造函数,创建一个线程并启动它来执行给定的可调用对象和参数。
thread(thread&& other) noexcept移动构造函数,获取另一个 std::thread 对象的所有权。
thread& operator=(thread&& other) noexcept移动赋值运算符,获取另一个 std::thread 对象的所有权。
~thread()析构函数,如果线程可 join,则会调用 join(),否则调用 detach()

二、thread类的用法

std::thread 类是 C++ 标准库中用于线程创建和管理的类。以下是 std::thread 类的一些关键用法和示例:

1. 创建线程

要创建一个线程,你需要实例化 std::thread 对象并传递一个函数或可调用对象(如 lambda 表达式或函数对象)作为参数。

#include <iostream>
#include <thread>

void threadFunction() {
    std::cout << "Hello from thread!" << std::endl;
}

int main() {
    std::thread t(threadFunction);
    t.join(); // 等待线程结束
    return 0;
}

2. 使用 Lambda 表达式

Lambda 表达式提供了一种简洁的方式来定义匿名函数对象,非常适合用于线程。

#include <iostream>
#include <thread>

int main() {
    std::thread t([]() {
        std::cout << "Hello from lambda thread!" << std::endl;
    });
    t.join();
    return 0;
}

3. 传递参数给线程

你可以将参数传递给线程函数。

void threadFunctionWithArgs(int x, double y) {
    std::cout << "x: " << x << ", y: " << y << std::endl;
}

int main() {
    std::thread t(threadFunctionWithArgs, 10, 3.14);
    t.join();
    return 0;
}

4. 线程的 join 和 detach

  • join(): 调用此方法会阻塞,直到线程结束执行。
  • detach(): 调用此方法会使线程在后台继续运行,而不受 std::thread 对象的生命周期限制。
int main() {
    std::thread t([]() {
        // 线程执行的代码
    });
    t.detach(); // 线程现在在后台运行,不会等待它结束
    // 主线程继续执行,而 t 线程在后台运行
    return 0;
}

5. 检查线程是否可 join

在调用 join()detach() 之前,可以使用 joinable() 检查线程是否还在运行。

int main() {
    std::thread t([]() {
        // 线程执行的代码
    });
    if (t.joinable()) {
        t.join(); // 线程还在运行,等待结束
    }
    return 0;
}

6. 线程的 ID

每个线程都有一个唯一的 ID,可以使用 get_id() 获取。

int main() {
    std::thread t([]() {
        std::cout << "Thread ID: " << std::this_thread::get_id() << std::endl;
    });
    t.get_id(); // 获取线程 ID
    t.join();
    return 0;
}

7. 线程的移动语义

std::thread 对象可以被移动,但不能被复制。这意味着你可以通过移动语义来转移线程的所有权。

int main() {
    std::thread t1([]() {
        std::cout << "Thread 1 running" << std::endl;
    });
    std::thread t2 = std::move(t1); // t1 的所有权转移给 t2
    t2.join(); // 等待线程结束
    return 0;
}

8. 线程的析构

std::thread 对象被销毁时,如果线程是可 join 的,那么 join() 会被自动调用。如果线程已经被分离,则不会有任何操作。

{
    std::thread t([]() {
        // 线程执行的代码
    });
    // t 在这里离开作用域,自动调用 join()
} // t 的析构函数被调用

🚨 注意事项

  • 确保在 std::thread 对象生命周期结束前,线程已经被正确处理(join 或 detach)。
  • 避免在线程函数中调用 std::exit() 或者抛出未捕获的异常,因为这可能会导致程序的不稳定。
  • 使用互斥锁和条件变量来同步线程,避免数据竞争和死锁。

std::thread 提供了强大的工具来实现多线程编程,但也需要谨慎使用以确保程序的正确性和稳定性

三、线程函数参数

线程函数的参数是以值拷贝的方式拷贝到线程栈空间中的,因此:即使线程参数为引用类型,在线程中修改后也不能修改外部实参,因为其实际引用的是线程栈中的拷贝,而不是外部实参

#include <thread>

void ThreadFunc1(int& x) {
    x += 10; 
}

void ThreadFunc2(int* x) {
    *x += 10; 
}

int main() {
    int a = 10;
	 // 在线程函数中对a修改,不会影响外部实参,因为:线程函数参数虽然是引用方式,但其实际引用的是线程栈中的拷贝    
    std::thread t1(ThreadFunc1, a);
    t1.join(); 
    
    // 如果想要通过形参改变外部实参时,必须借助std::ref()函数
    std::thread t2(ThreadFunc1, std::ref(a));
    t2.join(); // 等待线程执行完成
    
    // 地址的拷贝
    std::thread t3(ThreadFunc2, &a);
    t3.join(); // 等待线程执行完成   
    return 0;
}

温馨提示

感谢您对博主文章的关注与支持!另外,我计划在未来的更新中持续探讨与本文相关的内容,会为您带来更多关于C++以及编程技术问题的深入解析、应用案例和趣味玩法等。请继续关注博主的更新,不要错过任何精彩内容!

再次感谢您的支持和关注。期待与您建立更紧密的互动,共同探索C++、算法和编程的奥秘。祝您生活愉快,排便顺畅!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

代码随想录第二十三天 | 530.二叉搜索树的最小绝对差,501.二叉搜索树中的众数,236.二叉树的最近公共祖先

530.二叉搜索树的最小绝对差 看完想法&#xff1a;需要熟悉一下双指针的操作&#xff0c;好久没复习了&#xff0c;优先掌握递归 遇到在二叉搜索树上求什么最值&#xff0c;求差值之类的&#xff0c;都要思考一下二叉搜索树可是有序的&#xff0c;要利用好这一特点&#xff0…

el-date-picker选择开始日期的近半年

<el-date-pickerv-model"form[val.key]":type"val.datePickerType || daterange":clearable"val.clearable && true"range-separator"~"start-placeholder"开始日期"end-placeholder"结束日期"style&q…

ETF期权开户流程复杂吗?

ETF期权开户流程复杂吗&#xff1f;对于许多初次接触ETF期权的投资者来说&#xff0c;这个问题可能会让他们感到困惑。实际上&#xff0c;ETF期权开户的流程虽然涉及一些步骤&#xff0c;但只要遵循正确的指引&#xff0c;理解每一步的要求&#xff0c;整个过程并不会显得过于复…

【图像处理与机器视觉】频率域滤波

知识铺垫 复数 CRjI 可以看作复平面上的点&#xff0c;则该复数的坐标为&#xff08;R&#xff0c;I&#xff09; 欧拉公式 e j θ c o s θ j s i n θ e^{j\theta} cos \theta j sin \theta ejθcosθjsinθ 极坐标系中复数可以表示为&#xff1a; C ∣ C ∣ ( c o s…

安徽京准NTP时钟系统:GPS北斗卫星授时下的生活重塑

安徽京准NTP时钟系统&#xff1a;GPS北斗卫星授时下的生活重塑 安徽京准NTP时钟系统&#xff1a;GPS北斗卫星授时下的生活重塑 时间的流逝自古以来时钟都是人类生活与活动的基础。然而&#xff0c;随着科技的进步&#xff0c;我们对时间管理和测量的方法已经发生了翻天覆地的变…

0603《哎选》已经稳定运行2年

0603《哎选》已经稳定运行2年 0603《哎选》已经稳定运行2年 介绍 2022年6月3日经过一年的努力&#xff0c;优雅草蜻蜓G系统原生版诞生&#xff0c;本产品应用于《哎选》&#xff0c;经过2年的运营不断的更新迭代&#xff0c;目前产品已经有了一定的用户量&#xff0c;本产品…

毕业论文轻松写:AI写作工具如何助你一臂之力?

时间过的好快&#xff0c;马上又到了一年一度的毕业季了&#xff0c;对于即将毕业的学生来说毕业论文是一道难过的坎&#xff0c;想到自己为了毕业论文熬的夜&#xff0c;掉的头发&#xff0c;真的深有感触。 不过虽然翟博士给大家的毕业论文设了高门槛&#xff0c;但是随着时…

深度神经网络——什么是梯度下降?

如果对神经网络的训练有所了解&#xff0c;那么很可能已经听说过“梯度下降”这一术语。梯度下降是提升神经网络性能、降低其误差率的主要技术手段。然而&#xff0c;对于机器学习新手来说&#xff0c;梯度下降的概念可能稍显晦涩。本文旨在帮助您直观理解梯度下降的工作原理。…

【C#】类和结构体的区别

目录 1.区别概述 ​编辑 2.细节区别 3.结构体的特别之处 4.如何选择结构体和类 1.区别概述 结构体和类的最大区别是在存储空间上&#xff0c;前者是值类型&#xff0c;存储在栈上&#xff0c;后者是引用类型&#xff0c;存储在堆上&#xff0c;它们在赋值上有很大的区别&a…

Windows系统下安装JMeter

大家好&#xff0c;性能测试是现代软件开发中至关重要的一环&#xff0c;它能够帮助开发人员评估系统在不同负载条件下的稳定性和性能表现。而Apache JMeter作为一款功能强大的性能测试工具&#xff0c;广泛被业界采用。如果您正在Windows系统下寻求一种可靠的性能测试工具&…

引领未来,ArmSoM-Sige5震撼发布:RK3576芯片搭载,多媒体应用新宠

在数字化浪潮的推动下&#xff0c;ArmSoM-Sige5携手Rockchip RK3576第二代8纳米高性能AIOT平台&#xff0c;以颠覆性的性能和多功能性&#xff0c;成为多媒体应用的新宠儿。这一全新产品不仅拥有6 TOPS算力NPU和最大可配16GB大内存&#xff0c;更支持4K视频编解码&#xff0c;具…

Yuan 2.0-M32 是一个基于 Yuan 2.0 架构的双语混合专家 (MoE) 语言模型,旨在以更少的参数和计算量实现更高的准确率

主要创新点&#xff1a; 注意力路由器 (Attention Router): 提出了一种新的路由器网络&#xff0c;考虑了专家之间的相关性&#xff0c;从而提高了模型的准确率。高效计算&#xff1a; 使用 MoE 架构&#xff0c;40B 总参数中仅有 3.7B 激活参数&#xff0c;训练计算消耗仅为同…

串口控制小车和小车PWM调速

1.串口控制小车 1. 串口分文件编程进行代码整合&#xff0c;通过现象来改代码 2.接入蓝牙模块&#xff0c;通过蓝牙控制小车 3.添加点动控制&#xff0c;如果APP支持按下一直发数据&#xff0c;松开就停止发数据&#xff08;蓝牙调试助手的自定义按键不能实现&#xff09;&…

fastadmin批量导入

表的字段必须备注清楚导出的excel表头必须对应上如果mysql表有约束&#xff0c;导入会自动限制&#xff0c;挺方便的一个功能。

STM32-14-FSMC_LCD

STM32-01-认识单片机 STM32-02-基础知识 STM32-03-HAL库 STM32-04-时钟树 STM32-05-SYSTEM文件夹 STM32-06-GPIO STM32-07-外部中断 STM32-08-串口 STM32-09-IWDG和WWDG STM32-10-定时器 STM32-11-电容触摸按键 STM32-12-OLED模块 STM32-13-MPU 文章目录 1. 显示器分类2. LCD简…

R语言探索与分析-股票题目

Value at Risk&#xff08;VaR&#xff09;是一种统计技术&#xff0c;用于量化投资组合在正常市场条件下可能遭受的最大潜在损失。它是风险管理和金融领域中一个非常重要的概念。VaR通常以货币单位表示&#xff0c;用于估计在给定的置信水平和特定时间范围内&#xff0c;投资组…

深度剖析云边对接技术:探索开放API接口的价值与意义

在当今数字化时代的浪潮中&#xff0c;云边对接与开放API接口成为了塑造行业生态的重要驱动力。随着云计算、物联网和边缘计算等技术的快速发展&#xff0c;传统产业正在迈向数字化转型的关键时刻。而在这个过程中&#xff0c;云边对接技术以及开放的应用程序接口(API)扮演着举…

最新张量补全论文收集【8篇】

目录 1、利用张量子空间先验&#xff1a;增强张量补全的核范数最小化和 2、基于可学习空间光谱变换的张量核范数多维视觉数据恢复 3、用于图像补全的增强型低秩和稀疏 Tucker 分解 4、多模态核心张量分解及其在低秩张量补全中的应用 5、 低秩张量环的噪声张量补全 6、 视…

MYSQL ORDER BY

在MySQL中&#xff0c;默认情况下&#xff0c;升序排序会将NULL值放在前面&#xff0c;因为在排序过程中&#xff0c;NULL会被视为最小值。然而&#xff0c;有时会要求在升序排序中需要将NULL值放在最后。 例如根据日期升序时就会出现这种问题 方案一&#xff1a; SELECT sor…

微服务学习Day8-Sentinel

文章目录 Sentinel雪崩问题服务保护框架Sentinel配置 限流规则快速入门流控模式流控效果热点参数限流 隔离和降级FeignClient整合Sentinel线程隔离&#xff08;舱壁模式&#xff09;熔断降级 授权规则及规则持久化授权规则自定义异常结果持久化 Sentinel 雪崩问题 服务保护框架…