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

【C++多线程编程:六种锁】

目录

普通互斥锁: 

轻量级锁

独占锁:

std::lock_guard:

std::unique_lock: 

共享锁:

超时的互斥锁

递归锁


普通互斥锁: 

std::mutex确保任意时刻只有一个线程可以访问共享资源,在多线程中常用于保护共享资源。

互斥锁的实现示例:

#include <iostream>
#include <thread>
#include <mutex>

std::mutex Mutex;
int shared_source = 1;

void Shared_data()
{
	//获取锁
	Mutex.lock();
	shared_source++;
	std::cout << "shared_source : " << shared_source <<" by thread :"<<std::this_thread::get_id()<< std::endl;
	//锁释放
	Mutex.unlock();
}
int main()
{
	std::thread thread1(Shared_data);
	std::thread thread2(Shared_data);

	thread1.join();
	thread2.join();
	return 0;
}

在实际使用的过程中,如果容易忘记对锁进行释放,可以使用std::unique_lock和std::lock_guard安全的管理锁的释放。 

轻量级锁

在C++中,轻量级锁通常指的是一些开销较小,性能较高的锁机制,C++并没有直接提供“轻量级锁”的概念,可以通过一个自旋锁来达到实现轻量级锁的目的。

适用场景:竞争不激烈,比如在大多数时间里中有一个线程需要访问临界区的情况。但如果竞争激烈会导致CUP资源的浪费(CPU自旋),降低性能。

自旋锁:一种简单的锁机制,当线程尝试获取锁时,如果锁已被其他线程占用,则当前线程会不断地循环等待,直到锁可用

自旋锁的示例:

#include <iostream>
#include <thread>
#include <atomic>

std::atomic<bool>spinlock(false);

void spin_lock()
{
	while (spinlock.exchange(true, std::memory_order_acquire))
	{
		//如果锁被占用则一直自旋,等待锁可用
	}
}

void spin_unlock()
{
	spinlock.store(false, std::memory_order_release);//释放锁,并将锁的状态写入内存中
}
void critical_section()
{
	spin_lock();
	std::cout << "mutex acquire by thread : " << std::this_thread::get_id() << std::endl;
//	std::this_thread::sleep_for(std::chrono::seconds(1));
	spin_unlock();//锁释放
}
int main()
{
	std::thread t1(critical_section);
	std::thread t2(critical_section);

	t1.join();
	t2.join();

	return 0;
}

独占锁:

std::unique_lock和std::lock_guard都是cpp标准库提供的管理互斥锁的类,他们都提供了自动锁和解锁的功能。但二者存在一些关键区别:

std::lock_guard:

  • std::lock_guard并不是一种锁,而是一个作用域锁管理,一个能管理锁生命周期的工具,用于简化互斥锁的使用,确保在作用域结束后自动释放锁,避免死锁问题
  • 自动加锁和解锁:在构造时自动加锁,在析构时自动解锁,不需要显示的调用加锁和解锁的方法
  • 不支持条件变量配合使用:条件变量需要可以临时释放锁并重新获取锁,但是lock_guard并没有提供unlock().
  • 使用场景:适用于不需要显示控制加锁和解锁的场景,或者不需要与条件变量配合使用的场景,适用于在某个作用域中同步访问共享资源的场景

lock_guard示例:

#include <iostream>
#include <thread>
#include <mutex>

std::mutex mtx;
int shared_data = 0;

void worker()
{
	std::lock_guard<std::mutex>lock(mtx);//自动加锁
	shared_data++;
	std::cout << "shared_data ++ by thread : " << std::this_thread::get_id() << std::endl;
	//自动解锁,无需显示调用unlock()
}
int main()
{
	std::thread t1(worker);
	std::thread t2(worker);

	t1.join();
	t2.join();
	return 0;
}

std::unique_lock: 

  • std::unique_lock本身不是一种锁,而是一个可选的互斥锁管理器,提供了对互斥锁的更加灵活的控制方式
  • 显示加锁和解锁:提供了lock()和unlock()方法,可以显示的加锁和解锁,也可以在构造时自动加锁
  • 可以与条件变量配合
  • 自动管理锁的生命周期,避免了因忘记解锁而导致的死锁问题
  • 使用场景:需要显示控制加锁和解锁时机,或者需要和条件变量配合使用的场景

unique_lock实例:

#include <iostream>
#include <thread>
#include <mutex>

std::mutex mtx;
int shared_data = 0;
char str = 'a';

//不显示调用加解锁,自动加锁和解锁
void worker()
{
	std::unique_lock<std::mutex>lock(mtx);
	//defer_lock =>延迟锁定操作,需要在特定的时机才锁定互斥锁
	shared_data += 1;
	std::cout << "shared_data ++ :" << shared_data << " ,by thread : " << std::this_thread::get_id() << std::endl;
	std::cout << "str: " << str++ << " ,by thread: " << std::this_thread::get_id() << std::endl;
	
}
//显示调用加锁和解锁
void worker()
{
	std::unique_lock<std::mutex>lock(mtx,std::defer_lock);
	//defer_lock =>延迟锁定操作,需要在特定的时机才锁定互斥锁
	shared_data+=1;
	std::cout << "shared_data ++ :"<<shared_data<<" ,by thread : " << std::this_thread::get_id() << std::endl;
	lock.lock();//延迟锁定
	std::cout << "str: " << str++ <<" ,by thread: "<< std::this_thread::get_id() << std::endl;
	lock.unlock();//显示解锁
}

 int main()
{
	std::thread t1(worker);
	std::thread t2(worker);

	t1.join();
	t2.join();
	return 0;
}

独占锁与mutex相比,更加的安全,它可以避免忘记手动解锁,会在其作用域结束时自动的释放锁  

共享锁:

在C++中,共享锁允许多个线程共享读取资源,但在写入资源时只允许一个线程写入数据,要求独占锁。当某个线程获取了独占锁时,其他线程无法获取任何形式地锁,而当多个线程获取共享锁时,他们可以同时读取共享资源

使用场景:适用于读多写少的场景下,能有效提高多线程程序的性能

示例:

#include <iostream>
#include <thread>
#include <shared_mutex>

std::shared_mutex mtx;
int shared_data = 0;
void read_data()
{
	//获取共享锁
	std::shared_lock <std::shared_mutex> lock(mtx);
	std::cout << "Reading data : " << shared_data << std::endl;
}
void write_data()
{
	//获取独占锁
	std::unique_lock<std::shared_mutex>lock(mtx);
	shared_data = 24;
	std::cout << "Writing data : " << shared_data << std::endl;
}
int main()
{
	std::thread t1(read_data);
	std::thread t2(read_data);

	std::thread t3(write_data);


	t1.join();
	t2.join();
	t3.join();
	return 0;
}

超时的互斥锁

C++标准库提供了timed_mutex来实现该功能,支持超时机制,当线程尝试获取锁时,可以指定一个超时时间,如果在该时间内无法获取锁,线程将返回并继续执行其他任务。通过使用超时互斥锁,可以有效的避免线程在等待锁时无限地阻塞,提高程序地响应和稳定性 

示例: 

#include <iostream>
#include <thread>
#include <mutex>
#include <chrono>

//超时锁
//C++标准库提供的互斥锁之一,支持在尝试获取锁时设置超时时间
std::timed_mutex mtx;

void function_time()
{
	//在1s内获取锁
	//try_lock_for在指定时间内获取锁,成功返回true,失败则false
	//参数chrono是一个时间间隔类型的一个域
	if (mtx.try_lock_for(std::chrono::seconds(1)))
	{
		std::cout << "Lock acquired " << std::endl;

		//模拟耗时操作
		//在当前线程休眠这2s内,其他线程无法获取该锁
		//所以当t2尝试获取锁时,t1持有锁并休眠2s,当锁释放后,t2锁获取超时
		std::this_thread::sleep_for(std::chrono::seconds(2));
		mtx.unlock();//释放锁
	}
	else
	{
		std::cout << "Fail to acquire lock within 1s" << std::endl;
	}
}
int main()
{
	//主线程创建的这俩线程几乎同时开始执行function_time函数
	std::thread t1(function_time);
	std::thread t2(function_time);

	t1.join();
	t2.join();
	return 0;
}

递归锁

递归锁:同一个线程多次获取同一把锁,而不会导致死锁。cpp中没有直接提供递归锁的实现,但是可以通过reecursive_mutex来实现递归锁的功能。

reecursive_mutex:一个可重入的互斥锁,允许同一个线程多次调用lock或try_lock来获取锁,不会导致死锁。

使用场景:需要在同一个线程中多次获取锁的场景

递归锁示例:

#include <iostream>
#include <thread>
#include <mutex>

std::recursive_mutex mtx;

void function(int n)//递归
{
	if (n > 0)
	{
		mtx.lock();
		std::cout << "lock acquired by thread : " << std::this_thread::get_id() << " n ; " << n << std::endl;
		function(n - 1);
		mtx.unlock();
	}

}
 int main()
{
	std::thread t1(function,3);
	std::thread t2(function,2);

	t1.join();
	t2.join();
	return 0;
}


http://www.kler.cn/a/501516.html

相关文章:

  • 深度学习笔记11-优化器对比实验(Tensorflow)
  • python 寻找数据拐点
  • 面向对象分析与设计Python版 分析与设计概述
  • vue3后台系统动态路由实现
  • Termora 一个开源的 SSH 跨平台客户端工具
  • 该虚拟机似乎正在使用中。 如果该虚拟机未在使用,请按“获取所有权(T)”按钮获取它的所有权。否则,请按“取消(C)”按钮以防损坏。
  • 工作效率提升:使用Anaconda Prompt 创建虚拟环境总结
  • 基于Auto-Editor一键预处理音视频无声片段
  • 从零玩转CanMV-K230(9)-Timer、RTC、ADC、WDT、File
  • 介绍下不同语言的异常处理机制
  • Apache Hadoop YARN框架概述
  • 大模型LLM-Prompt-CRISPE
  • 文章题目:利用Adobe Flash Player漏洞:一次针对Windows XP的渗透测试实验
  • leetcode:1784. 检查二进制字符串字段(python3解法)
  • 贪心算法(五)
  • Node.js - 模块化与包管理工具
  • 苹果手机(IOS系统)出现安全延迟进行中如何关闭?
  • C#调用OpenCvSharp实现图像的膨胀和腐蚀
  • JavaScript动态渲染页面爬取之Splash
  • web前端第五次作业---制作菜单
  • 基于 SSH 的任务调度系统
  • 高可用技术:构筑数字世界的稳固根基
  • 26_Redis RDB持久化
  • Excel如何制作轮班表
  • Centos9 + Docker 安装 MySQL8.4.0 + 定时备份数据库到本地
  • 代码的形状:重构的方向