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

【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(); // 获取线程 IDt.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 的所有权转移给 t2t2.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++、算法和编程的奥秘。祝您生活愉快,排便顺畅!

相关文章:

  • 就凭这张图,下订华为享界S9
  • Linux系统使用Docker安装Drupal结合内网穿透实现远程访问管理后台
  • CMakeFile.txt通过sysroot方式后生成makefile报错
  • 构建LangChain应用程序的示例代码:7、如何使用Amazon Personalize服务的教程
  • git随记
  • ant design vue 表格错位,表头错位
  • 【安装笔记-20240528-Linux-在 Vultr 云服务器上安装 OpenWRT】
  • DP读书:《半导体物理学(第八版)》(七) 金属与半导体的接触- 10 min 速通(载流子分布)
  • vue项目路由跳转后上一页面未完成的接口取消请求
  • 视频汇聚管理平台EasyCVR程序报错“create jwtSecret del server class:0xf98b6040”的原因排查与解决
  • springboot基本使用十一(自定义全局异常处理器)
  • 【遂愿赠书 - 1期】:安恒“网安三剑客”-大模型时代下的网络安全实战指南
  • 学生信息管理系统C++
  • 插入排序与希尔排序
  • 生成式 AI——ChatGPT、Dall-E、Midjourney 等算法理念探讨
  • 【每日笔记】【Go学习笔记】2019-01-10 codis proxy处理流程
  • Debian下无root权限使用Python访问Oracle
  • JavaScript标准库系列——Math对象和Date对象(二)
  • k8s 面向应用开发者的基础命令
  • Laravel5.4 Queues队列学习
  • Mybatis初体验
  • nodejs实现webservice问题总结
  • Perseus-BERT——业内性能极致优化的BERT训练方案
  • Quartz初级教程
  • SpriteKit 技巧之添加背景图片
  • ucore操作系统实验笔记 - 重新理解中断
  • 使用 5W1H 写出高可读的 Git Commit Message
  • 微信小程序开发问题汇总
  • 交换综合实验一
  • 如何正确理解,内页权重高于首页?
  • ​数据链路层——流量控制可靠传输机制 ​
  • # 透过事物看本质的能力怎么培养?
  • ### Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLTr
  • (2/2) 为了理解 UWP 的启动流程,我从零开始创建了一个 UWP 程序
  • (Bean工厂的后处理器入门)学习Spring的第七天
  • (javaweb)Http协议
  • (初研) Sentence-embedding fine-tune notebook
  • (二) 初入MySQL 【数据库管理】
  • (四)React组件、useState、组件样式
  • (转)winform之ListView
  • .a文件和.so文件
  • .NET CF命令行调试器MDbg入门(一)
  • .net on S60 ---- Net60 1.1发布 支持VS2008以及新的特性
  • .net Signalr 使用笔记
  • .NET 应用架构指导 V2 学习笔记(一) 软件架构的关键原则
  • .netcore如何运行环境安装到Linux服务器
  • .NET版Word处理控件Aspose.words功能演示:在ASP.NET MVC中创建MS Word编辑器
  • .NET单元测试使用AutoFixture按需填充的方法总结
  • .net之微信企业号开发(一) 所使用的环境与工具以及准备工作
  • @property括号内属性讲解
  • @vue/cli脚手架
  • [ 2222 ]http://e.eqxiu.com/s/wJMf15Ku
  • [AutoSar]BSW_Com07 CAN报文接收流程的函数调用
  • [BZOJ 3531][Sdoi2014]旅行(树链剖分+线段树)
  • [C][栈帧]详细讲解