首页 > 编程语言 >C++中线程同步与互斥的四种方式介绍及对比详解

C++中线程同步与互斥的四种方式介绍及对比详解

时间:2025-01-16 16:25:08浏览次数:1  
标签:std mtx thread lock C++ 互斥 详解 线程

引言

在C++中,当两个或更多的线程需要访问共享数据时,就会出现线程安全问题。这是因为,如果没有适当的同步机制,一个线程可能在另一个线程还没有完成对数据的修改就开始访问数据,这将导致数据的不一致性和程序的不可预测性。为了解决这个问题,C++提供了多种线程同步和互斥的机制。

  1. 互斥量(Mutex)

互斥量是一种同步机制,用于防止多个线程同时访问共享资源。在C++中,可以使用std::mutex类来创建互斥量。

#include <thread>
#include <mutex>
 
std::mutex mtx; // 全局互斥量
int shared_data = 0; // 共享数据
 
void thread_func() {
     for (int i = 0; i < 10000; ++i) {
         mtx.lock(); // 获取互斥量的所有权
         ++shared_data; // 修改共享数据
         mtx.unlock(); // 释放互斥量的所有权
     }
}
 
int main() {
     std::thread t1(thread_func);
     std::thread t2(thread_func);
 
     t1.join();
     t2.join();
 
     std::cout << shared_data << std::endl; // 输出20000
 
     return 0;
}

在上述代码中,我们创建了一个全局互斥量mtx和一个共享数据shared_data。然后,我们在thread_func函数中使用mtx.lock()和mtx.unlock()来保护对shared_data的访问,确保在任何时候只有一个线程可以修改shared_data。

  1. 锁(Lock)

除了直接使用互斥量,C++还提供了std::lock_guard和std::unique_lock两种锁,用于自动管理互斥量的所有权。

#include <thread>
#include <mutex>
 
std::mutex mtx; // 全局互斥量
int shared_data = 0; // 共享数据
 
void thread_func() {
     for (int i = 0; i < 10000; ++i) {
         std::lock_guard<std::mutex> lock(mtx); // 创建锁,自动获取互斥量的所有权
         ++shared_data; // 修改共享数据
         // 锁在离开作用域时自动释放互斥量的所有权
     }
}
 
int main() {
     std::thread t1(thread_func);
     std::thread t2(thread_func);
 
     t1.join();
     t2.join();
 
     std::cout << shared_data << std::endl; // 输出20000
 
     return 0;
}

在上述代码中,我们使用std::lock_guard来自动管理互斥量的所有权。当创建std::lock_guard对象时,它会自动获取互斥量的所有权,当std::lock_guard对象离开作用域时,它会自动释放互斥量的所有权。这样,我们就不需要手动调用mtx.lock()和mtx.unlock(),可以避免因忘记释放互斥量而导致的死锁。

  1. 条件变量(Condition Variable)

条件变量是一种同步机制,用于在多个线程之间同步条件的变化。在C++中,可以使用std::condition_variable类来创建条件变量。

#include <thread>
#include <mutex>
#include <condition_variable>
 
std::mutex mtx; // 全局互斥量
std::condition_variable cv; // 全局条件变量
bool ready = false; // 共享条件
 
void print_id(int id) {
     std::unique_lock<std::mutex> lock(mtx); // 创建锁,自动获取互斥量的所有权
     while (!ready) { // 如果条件不满足
         cv.wait(lock); // 等待条件变量的通知
     }
     // 当收到条件变量的通知,且条件满足时,继续执行
     std::cout << "thread " << id << '\n';
}
 
void go() {
     std::unique_lock<std::mutex> lock(mtx); // 创建锁,自动获取互斥量的所有权
     ready = true; // 修改共享条件
     cv.notify_all(); // 通知所有等待的线程
}
 
int main() {
     std::thread threads[10];
     for (int i = 0; i < 10; ++i)
         threads[i] = std::thread(print_id, i);
 
     std::cout << "10 threads ready to race...\n";
     go(); // 开始比赛
 
     for (auto& th : threads) th.join();
 
     return 0;
}

在上述代码中,我们创建了一个全局互斥量mtx、一个全局条件变量cv和一个共享条件ready。然后,我们在print_id函数中使用cv.wait(lock)来等待条件变量的通知,当收到条件变量的通知,且条件满足时,继续执行。在go函数中,我们修改共享条件,并使用cv.notify_all()来通知所有等待的线程。

  1. 原子操作(Atomic Operation)

原子操作是一种特殊的操作,它可以在多线程环境中安全地对数据进行读写,而无需使用互斥量或锁。在C++中,可以使用std::atomic模板类来创建原子类型。

#include <thread>
#include <atomic>
 
std::atomic<int> shared_data(0); // 共享数据
 
void thread_func() {
     for (int i = 0; i < 10000; ++i) {
         ++shared_data; // 原子操作
     }
}
 
int main() {
     std::thread t1(thread_func);
     std::thread t2(thread_func);
 
     t1.join();
     t2.join();
 
     std::cout << shared_data << std::endl; // 输出20000
 
     return 0;
}

在上述代码中,我们创建了一个原子类型的共享数据shared_data。然后,我们在thread_func函数中使用++shared_data来进行原子操作,这样,我们就不需要使用互斥量或锁,也可以保证在任何时候只有一个线程可以修改shared_data。

  1. 对比
    策略 优点 缺点
    单一全局互斥量 简单 可能导致严重的性能问题,降低并发性
    多个互斥量 提高并发性 增加程序复杂性,需要避免死锁
    原子操作 提高并发性,避免互斥量开销 增加程序复杂性,需要理解和使用原子操作
    读写锁 提高并发性,特别是读操作多于写操作时 增加程序复杂性,需要管理读写锁,需要避免死锁

案例举例

假设我们正在开发一个在线聊天 服务器,需要处理大量的并发连接。每个连接都有一个关联的用户对象,用户对象包含了用户的状态信息,如用户名、在线状态等。

在这种情况下,我们可以使用多个互斥量的策略。我们可以将用户对象划分为几个组,每个组有一个关联的互斥量。当一个线程需要访问一个用户对象时,它只需要锁定该用户对象所在组的互斥量,而不是所有的用户对象。这样,不同的线程可以同时访问不同的用户对象,从而提高并发性。

同时,我们也可以使用读写锁的策略。因为在大多数情况下,线程只需要读取用户的状态信息,而不需要修改。所以,我们可以使用读写锁,允许多个线程同时读取用户对象,但在修改用户对象时需要独占锁。

在实践中,我们可能需要结合使用这两种策略,以达到最佳的效果。

  1. 更进一步:原子操作+锁

原子操作和锁是两种不同的线程同步机制,它们可以单独使用,也可以一起使用,具体取决于你的应用场景。

原子操作是一种低级的同步机制,它可以保证对单个内存位置的读写操作是原子的,即在任何时候只有一个线程可以对内存位置进行操作。原子操作通常用于实现高级的同步机制,如锁和条件变量。

锁是一种高级的同步机制,它可以保证对一段代码或多个内存位置的访问是原子的,即在任何时候只有一个线程可以执行被锁保护的代码或访问被锁保护的内存位置。

如果你在使用锁的同时还使用原子操作,那么你需要确保你的代码正确地理解和使用这两种同步机制。例如,如果你在一个被锁保护的代码段中使用原子操作,那么你需要确保原子操作不会违反锁的语义,即在任何时候只有一个线程可以执行被锁保护的代码。

以下是一个使用原子操作和锁的例子:

#include <thread>
#include <mutex>
#include <atomic>
 
std::mutex mtx; // 全局互斥量
std::atomic<int> counter(0); // 原子计数器
 
void thread_func() {
     for (int i = 0; i < 10000; ++i) {
         std::lock_guard<std::mutex> lock(mtx); // 获取互斥量的所有权
         ++counter; // 原子操作
         // 锁在离开作用域时自动释放互斥量的所有权
     }
}
 
int main() {
     std::thread t1(thread_func);
     std::thread t2(thread_func);
 
     t1.join();
     t2.join();
 
     std::cout << counter << std::endl; // 输出20000
 
     return 0;
}

在上述代码中,我们使用std::lock_guard来获取互斥量的所有权,然后使用++counter来进行原子操作。这样,我们既保证了在任何时候只有一个线程可以执行被锁保护的代码,也保证了对counter的操作是原子的。

总的来说,原子操作和锁可以一起使用,但你需要确保你的代码正确地理解和使用这两种同步机制。

总结

在C++中,当两个或更多的线程需要访问共享数据时,可以使用互斥量、锁、条件变量和原子操作等多种线程同步和互斥的机制来保证线程安全。选择哪种机制,取决于具体的应用场景和需求。

标签:std,mtx,thread,lock,C++,互斥,详解,线程
From: https://www.cnblogs.com/Ryan9399/p/18675192

相关文章

  • C++ open()和read()函数使用详解
    对于Framework工程师来说,必要C或者C++编程能力是必须的,像对设备节点的操作是最基本的操作,那么我们便会用到open和read函数。open()函数用于打开文件,而read()函数用于从打开的文件中读取数据。open()函数open()函数是C/C++标准库中的一个POSIX标准函数,用于打开一个文件并返回......
  • 深入理解C++ 空类大小
    在C++中,规定空类(即类中没有任何数据成员、成员函数、虚函数等成员的类)的大小为1字节,这背后主要有以下几方面的原因:保证对象的唯一性和可区分性在C++的面向对象编程模型中,对象是类的实例化结果,每个对象在内存中都需要占据一定的空间,以便程序能够通过地址等方式对其进行操作和区......
  • C++17 Filesystem 实用教程
    C++17标准带来了std::filesystem库,提供了强大的工具来处理文件路径,目录以及其他与文件系统相关的操作.这篇文章适合C++初学者以及希望掌握C++17新特性的开发者,旨在帮助他们高效地完成文件系统相关任务.什么是std::filesystem?std::filesystem是C++标准库的一部......
  • 【C++】开源:ImGui图形用户界面库配置与使用
    项目介绍项目Github地址:https://github.com/ocornut/imguiDearImGui(ImGui)是一个开源的、用C++编写的图形用户界面(GUI)库。它由OCornut创建,旨在为应用程序和工具提供创建用户界面的简单高效的方式。以下是DearImGui的一些主要特性和特点:1.即时模式GUI:ImGui遵循即......
  • c++基础算法讲解(写了ccf考试中可能出现的各种算法)
    枚举法枚举法是一种基本的问题解决策略,它尝试所有可能的情况以找到解决方案。这种方法通常用于问题规模较小且可以接受一定时间复杂度的情况。例子:找出三个数中最大的数#include<iostream>usingnamespacestd;intfindMax(inta,intb,intc){returnmax(a,......
  • c语言随机数rand与srand用法详解
    源文件test.c:#include<stdio.h>#include<stdlib.h>#include<time.h>intmain(){//生成并打印5个随机数for(inti=0;i<5;i++){//生成随机数并打印printf("第%d个随机数:%d\n",i+1,rand());}return0;}注意点:1)当不用srand()函数设置种子时,系统......
  • 访问者模式详解
    访问者模式(VisitorPattern)是一种行为型设计模式,它允许在不修改现有类结构的情况下,向现有类添加新的操作。该模式通过定义一个访问者接口,将算法与对象结构分离,使得操作可以独立于数据结构而变化。访问者模式的结构访问者模式主要包含以下组件:1. 抽象访问者(Visitor):• 声......
  • 空对象模式详解
    空对象模式详解定义空对象模式(NullObjectPattern)是一种行为设计模式,通过引入一个特殊的空对象来代替对空值的直接使用。这个空对象实现了与真实对象相同的接口,但其方法体通常为空或者提供默认的行为,从而避免了在代码中频繁进行空指针(NULL)检查。结构组成1. 抽象对象(Abst......
  • 编译原理实验四----NFA确定化(附C++代码)
    编译原理实验四----NFA确定化(附C++代码)经验分享算法思路前述知识点输入结构体子集法(确定化)代码1:寻找闭包代码2:自动机运作总流程代码3:重新命名最小化代码本文仅为编译原理课程实验记录开发过程,设计的知识点,以及实现算法的设计过程使用的是Qt开发......
  • 来试试用c++来测出你的幸运值吧~~~(1.0版)(while循环)(好玩小游戏)
    你是不是也想知道自己的幸运值呢?来试试看吧!废话不多说,上代码!不要忘记点赞哦~~~#include<bits/stdc++.h>#include<windows.h>usingnamespacestd;voidmeasureLuck(intl){intn;cout<<"输入测幸运次数吧!:";cin>>n;cout<<"最大的和是"<&......