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

C++_并发编程_thread_01_基本应用

  • 👋 Hi, I’m liubo
  • 👀 I’m interested in harmony
  • 🌱 I’m currently learning harmony
  • 💞️ I’m looking to collaborate on …
  • 📫 How to reach me …
  • 📇 sssssdsdsdsdsdsdasd
  • 🎃 dsdsdsdsdsddfsgdgasd
  • 🍺 jyukyuiyuiyuigkasd
  • 🍥 fsdfgdsgsdgdgadsa
  • ✨ xcvxcvxcvxcvdasdaasd
  • 🍰 dazdsxasxsaxsaasdsa
  • 🚨 gdfgdshdfhfhygjtyu

C++_并发编程_thread_01_基本应用

一、thread是什么

多线程是一种功能,它允许并发执行程序的两个或多个部分,以最大限度地利用 CPU。这种程序的每个部分都称为线程。多线程支持是在 C++11 中引入的。在 C++11 之前,我们必须使用 POSIX 线程或库。虽然这个库完成了这项工作,但缺乏任何标准语言提供的功能集导致了严重的可移植性问题。C++ 11 取消了所有这些,并给了我们 std::thread。线程类和相关函数在头文件<thread>中定义。

类thread表示单个执行线程。线程在构建关联的线程对象时立即开始执行。其定义用于观察和管理应用程序中的执行线程的对象。

包含头文件

#include <thread>

二、thread构造函数

std::thread 是 C++ 中表示单个线程的线程类。要启动线程,我们只需要创建一个新的线程对象,并将要调用的执行代码(即可调用对象)传递到对象的构造函数中。

1.默认构造函数

thread() noexcept

2.初始化构造函数

template <class Fn, class... Args>
explicit thread (Fn&& fn, Args&&... args);

该函数使用可变参数模板来构造一个线程对象,用来代表一个新的可join的执行线程。这个执行线程通过可变参数传入线程函数对象fn,以及函数的参数列表(可以简单理解为通过传值的方式将参数传给该构造函数)

3.拷贝构造函数

thread(thread&) = delete;
thread(const thread&) = delete;  // 禁用拷贝构造函数
thread(const thread&&) = delete;

线程不支持拷贝构造

4.移动构造

thread(thread&& __t) noexcept

转移参数x所代表的可执行指令的所有权,而不会影响线程的执行,转移后,参数x不再代表任何执行线程。

5.operator=

thread& operator=(const thread&) = delete; // 禁用拷贝赋值运算符
thread& operator=(thread&& __t) noexcept

thread对象不允许拷贝构造,同样的对于赋值操作符的重载实质是移动赋值。

三、thread可调用对象

创建 thread 对象后会将调用的执行代码(即可调用对象)传递到thread 对象的构造函数中。可调用对象可以是以下五个项中的任何一个:

  • 函数指针
  • Lambda 表达式
  • 函数对象
  • 非静态成员函数
  • 静态成员函数

定义可调用对象后,我们将其传递给 thread 构造函数。

1.使用函数指针启动线程

//函数指针可以是可调用对象,传递给 std::thread 构造函数以初始化线程。
void foo(param)
{ ... 
}
// The parameters to the function are put after the comma
std::thread thread_obj(foo, params);

2.使用 Lambda 表达式启动线程

//定义一个lambda表达式
auto f = [](params)
{...
};//使用 lambda 表达式作为可调用对象来启动
std::thread thread_obj(f, params);

3.使用函数对象启动线程

// 定义一个函数对象
class fn_object_class {// 重载operator()void operator()(params){ ...}
}std::thread thread_obj(fn_object_class(), params);

4.使用非静态成员函数启动线程

// 定义一个类
class Base {
public:// 非静态成员函数void foo(param) { ... }
}//创建Base类对象b
Base b;// 第一个参数是类非静态成员函数的引用
// 第二个参数类对象的引用
// 第三个参数是非静态成员函数的参数
std::thread thread_obj(&Base::foo, &b, params);

5.使用静态成员函数启动线程

// 定义一个类
class Base {
public://静态成员数static void foo(param) { ... }
}//创建Base类对象b
Base b;
// 其一个参数是类静态成员函数的引用
// 第二个参数是该函数的参数
std::thread thread_obj(&Base::foo, params);

四、join()

join()函数调用后主线程都要一直等待 thread 对象关联的线程执行完毕,然后再继续执行主线程或其他调用join()的线程。

行为:
同步执行:join()会阻塞调用它的线程,直到关联线程完成执行并结束
资源释放:join()等待关联线程结束后,会释放该线程的系统资源,包括线程栈和控制块等。这样可以防止资源泄露或孤立线程。
状态变化:join()后,std::thread对象的joinable()状态会变为false,防止再次join()

示例

#include <stdio.h>  // C语言的标准库,包含C语言流操作 printf等
#include <iostream> // 包含输入和输出操作
#include <string.h> // C语言的标准库,包含字符串处理操作 strcpy等
#include <unistd.h> // pause()头文件#include <thread>
#include <chrono>using std::cin;
using std::cout;
using std::endl;void pause_thread(int n)
{std::this_thread::sleep_for(std::chrono::seconds(n));std::cout << "暂停 " << n << " 秒结束\n";
}int main()
{printf("--------------------begain-------------------\n");cout << "Run Main Thread" << endl;std::cout << "生成三个线程...\n";std::thread t1(pause_thread, 1); // 线程这时候就已经开始启动std::thread t2(pause_thread, 2); // 线程这时候就已经开始启动std::thread t3(pause_thread, 3); // 线程这时候就已经开始启动std::cout << "线程已经生成,等待加入...:\n";t1.join();t2.join();t3.join();// 主线程等待三个线程结束后继续运行std::cout << "所有线程加入!\n";printf("--------------------end----------------------\n");// cin.get();// getchar();// pause();return EXIT_SUCCESS;
}

结果

--------------------begain-------------------
Run Main Thread
生成三个线程...
线程已经生成,等待加入...:暂停 1 秒结束
暂停 2 秒结束
暂停 3 秒结束
所有线程加入!
--------------------end----------------------

std::cout << “所有线程加入!\n”; 此词语句被阻塞等待三个线程执行完毕在执行.

五、detach()

detach()函数调用后会把主线程和新线程分离开,新线程的事情不影响主线程做事,后台自动回收。

行为:
分离状态:调用detach()后,std::thread对象与实际线程解绑,线程变为“分离状态”(detached state)。
资源管理:在分离状态下,线程在完成后会自行释放资源,不需要显示调用join()或其他操作。

下面的例子创建三个线程并分离,主线程等待三个线程5秒时间。

注意主线程结束会调用exit(),此函数将整个进程结束,所有的线程都会退出。

线程分离后不可接合并且可以安全地销毁。

示例

int main()
{printf("--------------------begain-------------------\n");cout << "Run Main Thread" << endl;std::cout << "生成三个线程...\n";std::thread t1(pause_thread, 1); // 线程这时候就已经开始启动std::thread t2(pause_thread, 2); // 线程这时候就已经开始启动std::thread t3(pause_thread, 3); // 线程这时候就已经开始启动std::cout << "线程已经生成,等待分离...:\n";t1.detach();t2.detach();t3.detach();std::cout << "所有线程分离!\n";// 给被分离线程5秒时间完成,但线程不一定完成!pause_thread(5); // 暂停5秒printf("--------------------end----------------------\n");// cin.get();// getchar();// pause();return EXIT_SUCCESS;
}

结果

--------------------begain-------------------
Run Main Thread
生成三个线程...
线程已经生成,等待分离...:
所有线程分离!暂停 1 秒结束
暂停 2 秒结束
暂停 3 秒结束
暂停 5 秒结束
--------------------end----------------------

std::cout << “所有线程分离!\n”; 词语句不会被阻塞,直接执行.

不同的系统有不同的现象:

  • windows:如果主线程退出了,那么系统会自动回收该进程的所有资源,包括分离出去的线程对象。
  • Linux:如果主线程退出了,被分离出去的线程对象会继续运行,结束时自动销毁线程资源。

六、joinable()

joinable()函数的作用是确认当前线程是否可以join(),如果joinable()函数的返回值为true,则该线程尚未通过join()终止或通过detach()分离。这意味着可以调用join()。这可以防止多次调用join()函数的错误。

示例

void pause_thread(int n)
{std::this_thread::sleep_for(std::chrono::seconds(n));std::cout << "暂停 " << n << " 秒结束\n";
}int main()
{printf("--------------------begain-------------------\n");cout << "Run Main Thread" << endl;std::cout << "生成三个线程...\n";std::thread t1(pause_thread, 1); // 线程这时候就已经开始启动std::thread t2(pause_thread, 3); // 线程这时候就已经开始启动cout << "joinable t1:" << t1.joinable() << endl;cout << "joinable t2:" << t2.joinable() << endl;cout << "t1.join:\n";t1.join();cout << "t2.detach:\n";t2.detach();cout << "joinable t1:" << t1.joinable() << endl;cout << "joinable t2:" << t2.joinable() << endl;pause_thread(5); // 暂停5秒printf("--------------------end----------------------\n");// cin.get();// getchar();// pause();return EXIT_SUCCESS;
}

结果

--------------------begain-------------------
Run Main Thread
生成三个线程...
joinable t1:1
joinable t2:1
t1.join:      // 阻塞一秒
暂停 1 秒结束
t2.detach:
joinable t1:0
joinable t2:0
暂停 3 秒结束
暂停 5 秒结束
--------------------end----------------------

七、总结

以上就是今天要讲的内容,后续会有更多内容。

八、参考资料

版权声明:本文参考了其他资料和CSDN博主的文章,遵循CC 4.0 BY-SA版权协议,现附上原文出处链接及本声明。

  1. https://blog.csdn.net/WangPaiFeiXingYuan/article/details/131022142
  2. https://blog.csdn.net/WangPaiFeiXingYuan/article/details/131022299
  3. https://blog.csdn.net/jianmo1993/article/details/134217076
  4. https://blog.csdn.net/qaaaaaaz/article/details/130794725
  5. https://blog.csdn.net/LeoLei8060/article/details/139476548




















相关文章:

C++_并发编程_thread_01_基本应用

&#x1f44b; Hi, I’m liubo&#x1f440; I’m interested in harmony&#x1f331; I’m currently learning harmony&#x1f49e;️ I’m looking to collaborate on …&#x1f4eb; How to reach me …&#x1f4c7; sssssdsdsdsdsdsdasd&#x1f383; dsdsdsdsdsddfsg…...

网络原理 - 4(TCP - 1)

目录 TCP 协议 TCP 协议段格式 可靠传输 几个 TCP 协议中的机制 1. 确认应答 2. 超时重传 完&#xff01; TCP 协议 TCP 全称为 “传输控制协议”&#xff08;Transmission Control Protocol&#xff09;&#xff0c;要对数据的传输进行一个详细的控制。 TCP 协议段格…...

强化学习框架:OpenRLHF源码解读,模型处理

本文主要介绍 强化学习框架&#xff1a;OpenRLHF源码解读&#xff0c;模型处理 models框架设计 了解一下 OpenRLHF的模型框架设计范式&#xff1a; From:https://arxiv.org/pdf/2405.11143 可以知道一个大概的流程&#xff1a;输入Pormpt通过Actor model输出回复 Response&am…...

STL常用算法——C++

1.概述 2.常用遍历算法 1.简介 2.for_each 方式一&#xff1a;传入普通函数&#xff08;printf1&#xff09; #include<stdio.h> using namespace std; #include<string> #include<vector> #include<functional> #include<algorithm> #include…...

UofTCTF-2025-web-复现

感兴趣朋友可以去我博客里看&#xff0c;画风更好看 UofTCTF-2025-web-复现 文章目录 scavenger-huntprismatic-blogscode-dbprepared-1prepared-2timeless scavenger-hunt 国外的一些ctf简单题就喜欢把flag藏在注释里&#xff0c;开源代码找到第一部分的flag 抓个包返回数据…...

Ruby 正则表达式

Ruby 正则表达式 引言 正则表达式&#xff08;Regular Expression&#xff0c;简称Regex&#xff09;是一种强大的文本处理工具&#xff0c;在编程和数据处理中有着广泛的应用。Ruby 作为一种动态、灵活的编程语言&#xff0c;同样内置了强大的正则表达式功能。本文将详细介绍…...

[密码学基础]GB与GM国密标准深度解析:定位、差异与协同发展

[密码学基础]GB与GM国密标准深度解析&#xff1a;定位、差异与协同发展 导语 在国产密码技术自主可控的浪潮下&#xff0c;GB&#xff08;国家标准&#xff09;与GM&#xff08;密码行业标准&#xff09;共同构建了我国商用密码的技术规范体系。二者在制定主体、法律效力、技术…...

代理设计模式:从底层原理到源代码 详解

代理设计模式&#xff08;Proxy Pattern&#xff09;是一种结构型设计模式&#xff0c;它通过创建一个代理对象来控制对目标对象的访问。代理对象充当客户端和目标对象之间的中介&#xff0c;允许在不修改目标对象的情况下添加额外的功能&#xff08;如权限控制、日志记录、延迟…...

15.第二阶段x64游戏实战-分析怪物血量(遍历周围)

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 本次游戏没法给 内容参考于&#xff1a;微尘网络安全 上一个内容&#xff1a;14.第二阶段x64游戏实战-分析人物的名字 如果想实现自动打怪&#xff0c;那肯定…...

HarmonyOS 基础语法概述 UI范式

ArkUI框架 - UI范式 ArkTS的基本组成 装饰器&#xff1a; 用于装饰类、结构、方法以及变量&#xff0c;并赋予其特殊的含义。如上述示例中Entry、Component和State都是装饰器&#xff0c;Component表示自定义组件&#xff0c;Entry表示该自定义组件为入口组件&#xff0c;Stat…...

专题讨论2:树与查找

在讨论前先回顾一下定义&#xff1a; BST树的定义 二叉搜索树是一种特殊的二叉树&#xff0c;对于树中的任意一个节点&#xff1a; 若它存在左子树&#xff0c;那么左子树中所有节点的值都小于该节点的值。 若它存在右子树&#xff0c;那么右子树中所有节点的值都大于该节点…...

django之数据的翻页和搜索功能

数据的翻页和搜素功能 目录 1.实现搜素功能 2.实现翻页功能 一、实现搜素功能 我们到bootstrap官网, 点击组件, 然后找到输入框组, 并点击作为额外元素的按钮。 我们需要使用上面红色框里面的组件, 就是搜素组件, 代码部分就是下面红色框框出来的部分。 把这里的代码复制…...

盈达科技GEO供应商:用AICC智能认知攻防系统重构AI时代的“内容主权”

《盈达科技GEO供应商&#xff1a;用AICC智能认知攻防系统重构AI时代的“内容主权”》 ——从全网认知统一到多模态智能投喂&#xff0c;破解生成式AI的内容暗战 前言 当用户向ChatGPT提问“XX品牌空调质量如何”时&#xff0c;AI的回答可能直接决定企业30%的潜在客户流向。 生…...

unity脚本-FBX自动化模型面数校验

根据目前模型资源平均面数预算进行脚本制作&#xff0c;自动化校验模型面数是否符合规范。 *注&#xff1a;文件格式为.cs。需要放置在unity资源文件夹Assets>Editor下。 测试效果&#xff08;拖一个fbx文件进unity时自动检测&#xff09;&#xff1a; 以下为完整代码 us…...

C++用于保留浮点数的两位小数,使用宏定义方法(可兼容低版本Visual Studio)

文章目录 一、 描述二、 样例二、 结果输出 一、 描述 这个宏定义&#xff08;可放入.h头文件里&#xff09;使用基本的数学运算&#xff0c;几乎兼容所有版本的VS&#xff0c;以下可对正数做四舍五入&#xff1a; #define ROUND_TO_TWO(x) ( (floor((x) * 100 0.5) / 100) …...

day30 学习笔记

文章目录 前言一、凸包特征检测1.穷举法2.QuickHull法 二、图像轮廓特征查找1.外接矩形2.最小外接矩形3.最小外接圆 前言 通过今天的学习&#xff0c;我掌握了OpenCV中有关凸包特征检测&#xff0c;图像轮廓特征查找的相关原理和操作 一、凸包特征检测 通俗的讲&#xff0c;凸…...

[密码学基础]密码学发展简史:从古典艺术到量子安全的演进

密码学发展简史&#xff1a;从古典艺术到量子安全的演进 密码学作为信息安全的基石&#xff0c;其发展贯穿人类文明史&#xff0c;从最初的文字游戏到量子时代的数学博弈&#xff0c;每一次变革都深刻影响着政治、军事、科技乃至日常生活。本文将以技术演进为主线&#xff0c;…...

(51单片机)LCD显示温度(DS18B20教程)(LCD1602教程)(延时函数教程)(单总线教程)

演示视频&#xff1a; LCD显示温度 源代码 如上图将9个文放在Keli5 中即可&#xff0c;然后烧录在单片机中就行了 烧录软件用的是STC-ISP&#xff0c;不知道怎么安装的可以去看江科大的视频&#xff1a; 【51单片机入门教程-2020版 程序全程纯手打 从零开始入门】https://www.…...

服务器运维:服务器流量的二八法则是什么意思?

文章目录 用户行为角度时间分布角度应用场景角度 服务器流量的二八法则&#xff0c;又称 80/20 法则&#xff0c;源自意大利经济学家帕累托提出的帕累托法则&#xff0c;该法则指出在很多情况下&#xff0c;80% 的结果是由 20% 的因素所决定的。在服务器流量领域&#xff0c;二…...

高并发秒杀使用RabbitMQ的优化思路

高并发秒杀使用RabbitMQ的优化思路 一、判断是否重复抢购&#xff08;防止一人多次秒杀)的逻辑1. 整体逻辑代码2. 原始判断重复抢购的方式&#xff1a;3. 后来优化为什么用 Redis 判断&#xff1f; 二、高并发下优化过的秒杀逻辑1.秒杀核心逻辑&#xff08;请求入口&#xff09…...

B + 树与 B 树的深度剖析

在数据库领域&#xff0c;B 树和 B 树是两种极为关键的数据结构&#xff0c;它们对于数据的存储、查询以及索引的构建等方面都有着深远的影响。深刻理解这两种树的原理、特性以及它们之间的差异&#xff0c;对于数据库的性能优化、数据组织和管理等工作具有不可替代的重要作用…...

【LeetCode】嚼烂热题100【持续更新】

2、字母异位词分组 方法一&#xff1a;排序哈希表 思路&#xff1a;对每个字符串排序&#xff0c;排序后的字符串作为键插入到哈希表中&#xff0c;值为List<String>形式存储单词原型&#xff0c;键为排序后的字符串。 Map<String, List<String>> m new Ha…...

赛灵思 XC7K325T-2FFG900I FPGA Xilinx Kintex‑7

XC7K325T-2FFG900I 是 Xilinx Kintex‑7 系列中一款工业级 (I) 高性能 FPGA&#xff0c;基于 28 nm HKMG HPL 工艺制程&#xff0c;核心电压标称 1.0 V&#xff0c;I/O 电压可在 0.97 V–1.03 V 之间灵活配置&#xff0c;并可在 –40 C 至 100 C 温度范围内稳定运行。该器件提供…...

【速写】多LoRA并行衍生的一些思考

迁移学习上的一个老问题&#xff0c;怎么做多领域的迁移&#xff1f;以前的逻辑认为领域迁移属于是对参数做方向性的调整&#xff0c;如果两个领域方向相左&#xff0c;实际上不管怎么加权相加都是不合理的。 目前一些做法想着去观察LoRA权重矩阵中的稠密块与稀疏块&#xff0…...

探索智能仓颉!Cangjie Magic:码字之间,意境自生

仓颉输入法&#xff0c;对于许多老牌中文使用者来说&#xff0c;不仅仅是一种输入工具&#xff0c;更是一种情怀&#xff0c;一种文化符号。它以拆字为核心&#xff0c;将汉字结构还原成最原始的构件&#xff0c;再通过特定的编码规则进行输入。然而&#xff0c;随着拼音输入法…...

py默认框架和代码

py默认框架 平常工作日常需要频繁写python脚本&#xff0c;留下一个常用的模板 # template.py import logging import json import time import functools import os from typing import Any, Dict, Optional, Union from pathlib import Path from logging.handlers import …...

通过 Samba 服务实现 Ubuntu 和 Windows 之间互传文件

在 Ubuntu 上进行配置 1. 安装 Samba 服务 打开终端&#xff0c;输入以下命令来安装 Samba&#xff1a; sudo apt update sudo apt install samba2. 创建共享目录 可以使用以下命令创建一个新的共享目录&#xff0c;例如创建名为 shared_folder 的目录&#xff1a; sudo m…...

k8s-1.28.10 安装metrics-server

1.简介 Metrics Server是一个集群范围的资源使用情况的数据聚合器。作为一个应用部署在集群中。Metric server从每个节点上KubeletAPI收集指标&#xff0c;通过Kubernetes聚合器注册在Master APIServer中。为集群提供Node、Pods资源利用率指标。 2.下载yaml文件 wget https:/…...

基于外部中中断机制,实现以下功能: 1.按键1,按下和释放后,点亮LED 2.按键2,按下和释放后,熄灭LED 3.按键3,按下和释放后,使得LED闪烁

题目&#xff1a; 参照外部中断的原理和代码示例,再结合之前已经实现的按键切换LED状态的实验&#xff0c;用外部中断改进其实现。 请自行参考文档《中断》当中&#xff0c;有关按键切换LED状态的内容, 自行连接电路图&#xff0c;基于外部中断机制&#xff0c;实现以下功能&am…...

【我的创作纪念日】 --- 与CSDN走过的第365天

个人主页&#xff1a;夜晚中的人海 不积跬步&#xff0c;无以至千里&#xff1b;不积小流&#xff0c;无以成江海。-《荀子》 文章目录 &#x1f389;一、机缘&#x1f680;二、收获&#x1f3a1;三、 日常⭐四、成就&#x1f3e0;五、憧憬 &#x1f389;一、机缘 光阴似箭&am…...