c++线程库操作

news2024/11/15 20:36:01

一、函数介绍

1、构造函数

无参构造函数:

thread thd = thread();

有参构造函数:

template<class Fn, class... Arg>

Fn:可调用对象(函数指针,仿函数,lambda表达式,包装器)

Arg:函数参数包

拷贝构造:

thread(const thread& th) = delete

不希望我们拷贝对象

移动构造:

thread(thread&& th)

支持移动构造

2、获取线程id

新线程因为有对象,所以直接调用函数 get_id()

主线程没有对象,用 this_thread::get_id()

3、其他函数

(1)jionable()

线程是否还在执行,joinable代表的是一个正在执行中的线程。

(2)jion()

该函数调用后会阻塞住线程,当该线程结束后,主线程继续执行

(3)detach()

在创建线程对象后马上调用,用于把被创建线程与线程对象分离开,分离 的线程变为后台线程,创建的线程的"死活"就与主线程无关

4、代码案例

void print(int n)
{
	for (int i = 0; i < n; i++)
	{
		cout << i << endl;
	}
}


int main()
{
	thread t1(print, 10);
	thread t2(print, 15);
	cout << "t1.get_id(): " << t1.get_id() << endl;
	cout << "t2.get_id(): " << t2.get_id() << endl;
	
	t1.join();
	t2.join();

	cout << "this thread: " << this_thread::get_id() << endl;
	return 0;
}

运行结果:

二、认识锁

1、背景

int x = 0;
void print(int n)
{
	for (int i = 0; i < n; i++)
	{
		x++; 
	}
}


int main()
{
	thread t1(print, 10000);
	thread t2(print, 15000);

	t1.join();
	t2.join();
	cout << x << endl;
	return 0;
}

预期结果应该是25000

这是因为多线程编程需要确保线程安全问题。

首先要明白线程拥有自己独立的栈结构,但对于全局变量等临界资源,是直接被多个线程共享的。这就会导致代码中的操作如果不是原子的线程(如 x++)安全就无法保障。

具体细节详见http://t.csdnimg.cn/zOevq

2、锁

(1)认识函数

引入锁的概念,如果一个线程竞争到锁,就会进入锁内部执行临界区代码,此时即使被切换,其他线程依然不会进入临界区,只能在临界区外等待锁被释放。

lock:竞争锁函数,如果竞争到锁就会返回,此时就可以向下执行代码。

unlock:解锁函数,执行完临界区代码解锁。

(2)代码案例

int x = 0;
mutex mtx;

void print(int n)
{
	//加锁建议在循环外
	mtx.lock();
	for (int i = 0; i < n; i++)
	{
		x++;
	}
	mtx.unlock();
}


int main()
{
	thread t1(print, 10000);
	thread t2(print, 15000);

	t1.join();
	t2.join();
	cout << x << endl;
	return 0;
}

运行结果:
 

由于加锁之后能保证操作的原子性,结果一定是准确的。

int main()
{
	int x = 0;
	mutex mtx;

	//先创建空的线程
	vector<thread> ths(3);

	auto func = [&](int n) {
		mtx.lock();
		for (int i = 0; i < n; i++)
		{
			x++;
		}
		mtx.unlock();
	};
	
	for (auto& th : ths)
	{
		//构造匿名对象,由于之前创建了空的线程,
		//这里调用 operator= 移动赋值
		th = thread(func, 10000);
	}

	for (auto& th : ths)
	{
		th.join();
	}
	cout << x << endl;
	return 0;
}

运行结果:
 

3、其他锁类型

(1)recursive_mutex

递归互斥锁,这把锁主要用来递归加锁的场景中,因为递归会引起死锁问题。

为什么会出现死锁?
因为当前在进入递归函数前,申请了锁资源,进入递归函数后(还没有释放锁资源),再次申请锁资源,此时就会出现锁在我手里,但我还申请不到的现象,也就是死锁。

解决这个 死锁 问题的关键在于 自己在持有锁资源的情况下,不必再申请,此时就要用到recursive_mutex

(2)timed_mutex

时间互斥锁,这把锁中新增了定时解锁的功能,可以在程序运行指定时间后,自动解锁(如果还没有解锁的话)

(3)recursive_time_mutex

递归时间互斥锁,就是对timed_mutex时间互斥锁做了递归方面的升级,使其在面对递归场景时,不会出现死锁。

(4)RAII风格锁

a、lock_guard

锁守卫,只用lock, unlock会导致抛异常后难以解决死锁问题,lock_guard是一个类模板,构造时加锁,析构时解锁,如果执行管理代码的一部分可以加局部域 {}

b、unique_lock

特殊锁,类似于lock_guard,但是支持手动加锁解锁。

4、原子操作

是一个类模板,里面封装了大部分内置类型的++ -- ^ 等原子性操作,这样可以不用加锁来实现内置类型的一些原子操作。

内置类型如下:

获取参数用 load 函数

int main()
{
	vector<thread> vthd;
	int n;
	cin >> n;
	vthd.resize(n);

	atomic<int> x = 0;

	mutex mtx;
	auto func = [&](int n) {
		//mtx.lock();
		// 局部域
		{
			//lock_guard<mutex> lock(mtx);
			for (size_t i = 0; i < n; i++)
			{
				++x;
			}
			//mtx.unlock();
		}
	};
	for (auto& thd : vthd)
	{
		// 移动赋值
		thd = thread(func, 100000);
	}

	for (auto& thd : vthd)
	{
		thd.join();
	}
	printf("%d\n", x.load());
	return 0;
}

运行结果:
 

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

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

相关文章

掌握 ERP 进销存系统源码,实现企业精准管理 带源代码包以及搭建部署教程

系统概述 ERP 进销存系统源码是一套基于先进技术架构开发的企业管理解决方案。它涵盖了企业采购、销售、库存管理等核心业务领域&#xff0c;通过信息化手段实现了数据的实时共享、流程的优化整合以及决策的科学支持。 该系统源码采用了模块化设计理念&#xff0c;各个模块之…

传输层(多路复用与解复用)

目录 1.概述传输层服务 传输服务和协议 传输层 VS 网络层 类比&#xff1a;两个家庭的通信 Internet传输层提供的服务 2.多路复用与解复用 多路复用/解复用 多路复用的工作原理 无连接&#xff08;UDP&#xff09;多路复用 UDP多路复用例子 UDP多路解复用例子 面向连…

【Python报错已解决】ValueError: cannot reindex from a duplicate axis

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 引言&#xff1a; 当处理Pandas数据框&#xff08;DataFrame&#xff09;时&#xff0c;你是否遇到过ValueError: cannot reind…

零知识证明-公钥分发方案DH((六)

前言 椭圆曲线配对&#xff0c;是各种加密构造方法(包括 确定性阀值签名、zk-SNARKs以及相似的零知识证明)的关键元素之一。椭圆曲线配对(也叫“双线性映射”)有了30年的应用历史&#xff0c;然而最近这些年才把它应用在密码学领域。配对带来了一种“加密乘法”的形式&#xff…

VLAN原理和配置

VLAN技术可以将一个物理局域网在逻辑上划分成多个广播域&#xff0c;也就是多个VLAN。VLAN技术部署在数据链路层&#xff0c;用于隔离二层流量。同一个VLAN内的主机共享同一个广播域&#xff0c;它们之间可以直接进行二层通信。 VLAN标签长4个字节&#xff0c;直接添加在以太网…

轻松享受远程办公:可道云teamOS,让自由与效率同行

职场生活中&#xff0c;我们常常会因为工作需要而面临出差的情况。在这种情况下&#xff0c;如何能与不在身边的公司同事组员&#xff0c;保持高效协作&#xff0c;就显得尤为重要了。 移动办公新体验 记得有一次&#xff0c;我正在外地参加一个重要的商务会议&#xff0c;突…

佰朔资本:8.87亿人次!全国铁路 暑运发送旅客创历史同期新高

记者1日从我国国家铁路集团有限公司得悉&#xff0c;8月31日&#xff0c;为期62天的铁路暑运圆满结束。7月1日至8月31日&#xff0c;全国铁路累计发送旅客8.87亿人次&#xff0c;同比增长6.7%&#xff0c;日均发送旅客1431.2万人次&#xff0c;创暑运旅客发送量前史新高&#x…

如何恢复图库里的照片?照片恢复有道,最后一招更有效!

在今天&#xff0c;手机里的照片不仅是记忆的载体&#xff0c;更是我们情感的寄托。然而&#xff0c;当我们在查看照片时不小心删除或丢失重要照片的情况时有发生&#xff0c;这可能会让我们感到后悔和焦虑。我们也会想&#xff1a;如何恢复图库里的照片呢&#xff1f;失去的照…

Upscayl 采用开源人工智能技术,可以增强低分辨率图像的效果。

Upscayl 是一款免费开源的基于 AI 神经网络与深度学习的「图片画质提升 / 超分辨率软件」&#xff0c;可以做到“无损放大图片”&#xff0c;让你轻松将任意分辨率的图片、照片、壁纸放大到高清、超清甚至 4K 水平&#xff0c;大幅提升图片细节表现与清晰度&#xff01;效果比起…

谷粒商城实战笔记-问题记录-Feign异步调用丢失请求头问题

文章目录 单线程下生效的原理多线程下Interceptor不生效的原因解决方案1&#xff0c;不优雅的方法2&#xff0c;优雅的方法 在请求多个信息时&#xff0c;我们使用了多线程&#xff0c;这就带来了一个问题&#xff0c;前面我们解决Feign丢失请求头的方案在多线程下&#xff0c;…

笔记本连wifi蓝屏问题修复

最近笔记win11专业版本连wifi时总是蓝屏重启&#xff0c;各种杀毒软件升级补丁都无效&#xff0c;最后升级了网卡驱动&#xff0c;就正常了 下载链接&#xff1a; 23.40.0.4无线网卡驱动 下载后直接运行&#xff0c;运行后网卡驱动版本变成 23.40.0.4 就好了&#xff01;&am…

Spring理论知识(Ⅲ)——Spring面向切面编程

Spring的组成 Spring由20个核心依赖组成&#xff0c;这20个核心依赖可以分为6个核心模块 本文主要讲解spring的AOP模块&#xff0c;其中包括spring-aop&#xff0c;spring-aspects Spring AOP模块概述 AOP思想 Spring AOP全面详解(超级详细) AOP 是一种编程思想&…

华为云低代码AstroZero技巧教学3:智能计算商品费用,轻松实现高效数据处理

公司经营过程中&#xff0c;多个场景会涉及到计算商品花费。以企业内部行政采购为例&#xff0c;在统计相关采购清单中&#xff0c;会涉及到诸多数据统计及计算。如采购商品种类、数量、单价以及其他附加成本&#xff08;运输费用&#xff0c;装卸费用&#xff0c;包装费用&…

网络安全ctf比赛/学习资源整理,解题工具、比赛时间、解题思路、实战靶场、学习路线,推荐收藏!

前言 对于想学习或者参加CTF比赛的朋友来说&#xff0c;CTF工具、练习靶场必不可少&#xff0c;今天给大家分享自己收藏的CTF资源&#xff0c;希望能对各位有所帮助。 CTF在线工具 首先给大家推荐我自己常用的3个CTF在线工具网站&#xff0c;内容齐全&#xff0c;收藏备用。…

第11章 第9节 基于软件质量特性的测试(软件评测师)

1.以下关于软件质量特性测试的叙述&#xff0c;正确的是&#xff08;&#xff09; 1.成熟性测试是检验软件系统故障&#xff0c;或违反指定接口的情况下维持规定的性能水平有关的测试工作 &#xff08;&#xff0c;成熟性测试是检验软件产品是否具备为避免由软件中错误而导致…

平价蓝牙耳机哪个音质好性价比高?四大超值平价机型推荐

在当今市场上&#xff0c;蓝牙耳机的选择众多&#xff0c;从高端品牌到平价型号&#xff0c;消费者面对的选择可谓是五花八门&#xff0c;并非每个人都愿意或能够负担得起昂贵的价格以获取顶级音质&#xff0c;所以平价蓝牙耳机哪个音质好性价比高&#xff1f;对于那些寻求性价…

springboot党员之家服务系统小程序论文源码调试讲解

第二章技术介绍 2.1B/S结构 随着软件系统的不断改进和升级&#xff0c;B/S结构产品更为方便的特征体现地十分明显。对于一个中等偏大的公司来说&#xff0c;如果系统管理员每天要在很多台电脑之间来回查看&#xff0c;不断奔走&#xff0c;那么效率和工作量就会变得很低&#…

服务器数据恢复—如何应对双循环RAID5阵列的数据丢失问题?

服务器存储数据恢复环境&#xff1a; 一台存储中有一组由7块硬盘组建的RAID5阵列&#xff0c;存储中还有另外3块盘是raid中掉线的硬盘&#xff08;硬盘掉线了&#xff0c;管理员只是添加一块的新的硬盘做rebuild&#xff0c;并没有将掉线的硬盘拔掉&#xff09;。整个RAID5阵列…

大模型全攻略:数据准备、模型微调到部署,一文全解析!

0. 引言 距离ChatGPT已经发布1年半了&#xff0c;距离我们训练出自己的大模型也已经1周年了。目前仍然有很多同学在咨询如何训练自己的大模型。这个东西和男/女朋友一样。当你不认识TA&#xff0c;距离TA很远&#xff0c;不敢接触TA的时候&#xff0c;TA就是很神秘&#xff0c…

基础闯关4

环境配置 我们来配置LlamaIndex实验环境&#xff0c;首先创建Python环境并安装必要的库&#xff1a; conda create -n llamaindex python3.10 conda activate llamaindex conda install pytorch2.0.1 torchvision0.15.2 torchaudio2.0.2 pytorch-cuda11.7 -c pytorch -c nvid…