C++:线程(thread)的创建、调用及销毁

在 C++ 中,线程的管理主要依赖于标准库 std::thread,自 C++11 起,这一功能被标准化,使得我们能够更加方便地创建、管理和销毁线程。这里我们详细讲解线程的创建、调用和销毁流程。

1. 线程的创建

创建线程通常是为了在单独的线程中执行某个任务。我们可以通过 std::thread 对象来创建一个新的线程。一个线程可以从以下几种类型的可调用对象启动:

  • 普通函数
  • Lambda 表达式
  • 函数对象
  • 类的成员函数

1.1 使用普通函数

#include <iostream>
#include <thread>void printMessage() {std::cout << "Hello from thread!" << std::endl;
}int main() {std::thread myThread(printMessage); // 创建一个线程并启动 printMessagemyThread.join();                    // 等待线程完成return 0;
}
  • std::thread myThread(printMessage) 创建一个线程对象 myThread,并启动执行 printMessage 函数。
  • myThread.join() 等待线程完成。如果没有调用 join()detach(),程序在结束时会崩溃。

1.2 使用 Lambda 表达式

#include <iostream>
#include <thread>int main() {std::thread myThread([]() {std::cout << "Hello from lambda!" << std::endl;});myThread.join();return 0;
}
  • 这里我们创建了一个线程并使用 lambda 表达式作为线程函数。Lambda 允许我们在局部作用域中定义线程任务。

1.3 使用函数对象

#include <iostream>
#include <thread>class PrintTask {
public:void operator()() const {std::cout << "Hello from function object!" << std::endl;}
};int main() {std::thread myThread(PrintTask()); // 创建线程并启动myThread.join();return 0;
}
  • 这是通过重载 operator() 来定义一个可调用的对象,该对象可以直接用来创建线程。

1.4 使用类的成员函数

#include <iostream>
#include <thread>class MyClass {
public:void memberFunction() {std::cout << "Hello from member function!" << std::endl;}
};int main() {MyClass obj;std::thread myThread(&MyClass::memberFunction, &obj); // 需要传递对象指针myThread.join();return 0;
}
  • 如果是成员函数,则需要传递对象指针。&MyClass::memberFunction 表示成员函数地址,&obj 是指向 MyClass 实例的指针。

2. 线程的调用

  • 线程的参数传递:你可以向线程函数传递参数,它们会按照值传递的方式进行复制。为了传递引用,可以使用 std::refstd::cref

#include <iostream>
#include <thread>void printNumber(int n) {std::cout << "Number: " << n << std::endl;
}int main() {int value = 42;std::thread myThread(printNumber, value);myThread.join(); // 等待线程完成return 0;
}
  • 捕获引用
void increment(int& n) {++n;
}int main() {int value = 0;std::thread myThread(increment, std::ref(value));myThread.join();std::cout << "Value after increment: " << value << std::endl;return 0;
}

这里 std::ref 确保 value 以引用的形式传递。

3. 线程的同步

  • join():主线程会等待 myThread 结束。join 是同步机制,用于确保线程完成后主线程才会继续。
std::thread myThread(task);
myThread.join();
  • detach():将线程从主线程分离,让它独立运行。独立运行的线程在后台执行,主线程不再等待它完成。需谨慎使用,可能引发访问冲突。
std::thread myThread(task);
myThread.detach();

4. 线程的销毁

  • std::thread 对象离开作用域时,如果没有调用 join()detach(),程序会触发异常终止。
  • 使用 join() 可以让主线程等待子线程完成,从而安全地销毁线程。
  • 使用 detach() 可以将线程从 std::thread 对象中分离,使其成为独立线程。调用 detach() 后,std::thread 对象不再管理该线程。

5. 线程的生命周期管理

  • RAII:考虑使用 RAII 类管理线程生命周期,确保在对象析构时 join()detach() 线程,从而避免泄漏和不正确的管理。
class ThreadGuard {
public:explicit ThreadGuard(std::thread& t) : thread(t) {}~ThreadGuard() {if (thread.joinable()) {thread.join();}}private:std::thread& thread;
};

6. 线程的注意事项

  • 避免数据竞争和同步问题:线程共享数据时要小心。可以使用 std::mutex 进行保护,或者使用其他同步机制如 std::lock_guard
  • 避免内存泄漏:如果使用 new thread(),确保 delete 以释放分配的内存。
  • 检查 joinable() 状态:调用 join()detach() 前,可以用 joinable() 检查线程状态。

 7. 示例总结

#include <iostream>
#include <thread>void task(int id) {std::cout << "Thread " << id << " is running." << std::endl;
}int main() {std::thread t1(task, 1); // 创建线程并传递参数std::thread t2(task, 2);t1.join(); // 等待 t1 完成t2.join(); // 等待 t2 完成return 0;
}

8. 线程管理总结

  • 使用 std::thread 创建和管理线程。
  • join()detach() 用于控制线程的生命周期。
  • 避免重复 join()detach(),确保资源管理得当。
  • 使用同步机制保护共享数据的访问。

通过这种方式,你可以更灵活地创建、管理和销毁 C++ 线程,确保程序的并发性和资源管理的安全性。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【LeetCode】【算法】64. 最小路径和

LeetCode 64. 最小路径和 题目描述 给定一个包含非负整数的 m x n 网格 grid &#xff0c;请找出一条从左上角到右下角的路径&#xff0c;使得路径上的数字总和为最小。 说明&#xff1a;每次只能向下或者向右移动一步。 思路 思路&#xff1a;这种题太典了&#xff0c;典…

1.7 JS性能优化

从输入url到页面加载完成都做了些什么 输入 URL - 资源定位符 http://www.zhaowa.com - http 协议 追问&#xff1a;http 与 TCP 1. http - 应用层 < > TCP - 传输层 2. 关联 - http基于TCP实现连接 < > UDP > 握手 & 挥手 &#xff08;传输速率上较…

Spring Task详细讲解

✨Spring Task简介 Spring Task 是 Spring 提供的轻量级定时任务工具&#xff0c;也就意味着不需要再添加第三方依赖了&#xff0c;相比其他第三方类库更加方便易用。可以按照约定的时间自动执行某个代码逻辑。 使用场景&#xff1a; 信用卡每月还款提醒银行贷款每月还款提醒…

Qt/C++ 海康SDK开发示例Demo

*** 工业相机在机器视觉中起到关键作用&#xff0c;本文基于海康 SDK 详细解读了设备连接与控制的各个步骤。内容涵盖设备枚举、句柄创建、图像采集回调以及设备异常处理&#xff0c;帮助开发者快速理解如何通过代码控制相机&#xff0c;实时采集并处理图像数据。*** 1. 搜索并…

HDLBIts习题(5):移位寄存器

&#xff08;1&#xff09;易错习题1&#xff1a;109题&#xff08;shift18&#xff09; 对算数左移和算数右移概念不清&#xff0c;不知道该如何计算。 逻辑左移和算术左移之间没有区别。&#xff08;无论是有符号位数据还是无符号位数据&#xff0c;右侧补0&#xff09; 逻辑…

想要成为独立游戏作者 :通关!游戏设计之道 2-2 关卡设计

本文通过ai辅助总结加个人微调,不喜勿喷 前篇如下&#xff1a; 想要成为独立游戏作者 &#xff1a;通关&#xff01;游戏设计之道 2-1 HUD-CSDN博客 1.关卡的多重定义 在电子游戏行业里 “关卡” 有多种含义&#xff0c;如游戏行为发生的环境、分割的游戏体验单元、量…

【深圳大学】数据结构A+攻略(计软版)

1. 考试 1.1 形式 分为平时&#xff0c;笔试&#xff0c;机试三部分。其中&#xff1a; 平时占30%&#xff0c;包含平时OJ测验和课堂练习&#xff0c;注意这个可能会因老师的不同和课题组的新策略而改变。笔试占60%&#xff0c;是分值占比的主要部分。机试占10%。 1.2 题型…

Springboot 启动端口占用如何解决

Springboot 启动端口占用如何解决 1、报错信息如下 *************************** APPLICATION FAILED TO START ***************************Description:Web server failed to start. Port 9010 was already in use.Action:Identify and stop the process thats listening o…

H.264/H.265播放器EasyPlayer.js RTSP播放器关于webcodecs硬解码H265的问题

EasyPlayer.js H5播放器&#xff0c;是一款能够同时支持HTTP、HTTP-FLV、HLS&#xff08;m3u8&#xff09;、WS视频直播与视频点播等多种协议&#xff0c;支持H.264、H.265、AAC、G711A、Mp3等多种音视频编码格式&#xff0c;支持MSE、WASM、WebCodec等多种解码方式&#xff0c…

集合类源码浅析のJDK1.8ConcurrentHashMap(上篇)

文章目录 前言一、概述二、CHM的属性1、属性 三、新增方法1、put2、initTable 四、分段计数1、addCount2、fullAddCount3、sumCount 总结 前言 本篇是JDK1.8的ConcurrentHashMap源码个人学习笔记&#xff0c;ConcurrentHashMap&#xff08;笔记中简称CHM&#xff09;是一种线程…

Linux权限和开发工具(3)

文章目录 1. 简单理解版本控制器Git1. 如何理解版本控制 2. Git的操作2.1 Git安装2.2 Git提交身份2.3 Git提交命令2.4 Git版本管理2.5 Git下的同步 3. gdb命令3.1解决gdb的难用问题3.2 gdb/cgdb的使用 1. 简单理解版本控制器Git 1. 如何理解版本控制 我们在做项目的时候可能会…

抓包工具WireShark使用记录

目录 网卡选择&#xff1a; 抓包流程&#xff1a; 捕获过滤器 常用捕获过滤器&#xff1a; 抓包数据的显示 显示过滤器&#xff1a; 常用的显示过滤器&#xff1a; 实际工作中&#xff0c;在平台对接&#xff0c;设备对接等常常需要调试接口&#xff0c;PostMan虽然可以进…

腾讯云双十一重磅福利----下一代CDN-EdgeOne

&#x1f34b;引言 随着全球互联网的快速发展和网络安全威胁的不断升级&#xff0c;传统的内容分发网络&#xff08;CDN&#xff09;已逐渐无法满足高效、安全、灵活的需求。腾讯云的下一代CDN产品—EdgeOne应运而生&#xff0c;凭借其全球化边缘节点架构&#xff0c;为客户提供…

Unity Coroutine

调用函数时&#xff0c;函数将运行到完成状态&#xff0c;然后返回。这实际上意味着在函数中发生的任何动作都必须在单帧更新内发生&#xff1b;函数调用不能用于包含程序性动画或随时间推移的一系列事件。例如&#xff0c;假设需要逐渐减少对象的 Alpha&#xff08;不透明度&a…

qt QGraphicsProxyWidget详解

1. 概述 QGraphicsProxyWidget 类是 Qt 图形视图框架中的一个关键类&#xff0c;它允许 QWidget 组件被嵌入到 QGraphicsScene 中。QGraphicsProxyWidget 作为一个代理&#xff0c;它在 QGraphicsScene 和 QWidget 之间建立了桥梁&#xff0c;使得 QWidget 可以在 QGraphicsVi…

【数据集】GPM IMERG Daily Precipitation Dataset

目录 数据集概述时空分辨率:数据下载参考数据集概述 Global Precipitation Measurement (GPM) IMERG Daily Precipitation Dataset 简介 Global Precipitation Measurement (GPM) 是由美国国家航空航天局(NASA)和日本宇宙航空研究开发机构(JAXA)联合发起的一项全球降水观…

MySQL 迁移 dm

参考链接 此处为语雀内容卡片&#xff0c;点击链接查看&#xff1a;MySQL 5.7.27 迁移 DM 8 语雀 迁移前准备 以下中的命名&#xff0c;密码都是可修改的&#xff0c;这里给出的就只是一个例子 创建表空间 # 创建表空间名为 dbTest&#xff0c;路径为 Z:\fei\data\dm\dbT…

新版IJidea 如何打开数据库窗口(2024.2.4 版)(连接数据库)

新版IJidea 2024.2.4 如何打开数据库窗口&#xff1f; 方式&#xff1a;使用插件&#xff0c;Database Navigator 1.安装插件&#xff0c;步骤如下&#xff1a; 打开 Settings/Preferences 对话框&#xff08;快捷键 CtrlAltS&#xff09;。前往 Plugins 菜单项。在搜索框中…

MySQL:left join后用on与where的区别

一、前言 前几天项目中&#xff0c;写SQL时本想通过 A left B join on and 后面的条件来使查出的两条记录变成一条&#xff0c;奈何发现还是有两条。在此记录一下&#xff0c;on与where的区别。 二、ON 原始数据展示 SELECT t1.*,t2.* FROM t_test_staff t1 left join t_te…

Spark 核心概念与宽窄依赖的详细解析

Spark 的介绍与搭建&#xff1a;从理论到实践_spark环境搭建-CSDN博客 Spark 的Standalone集群环境安装与测试-CSDN博客 PySpark 本地开发环境搭建与实践-CSDN博客 Spark 程序开发与提交&#xff1a;本地与集群模式全解析-CSDN博客 Spark on YARN&#xff1a;Spark集群模式…