C++11 同步与互斥

0. C++11的线程

#include <thread>

  • 面向对象的接口
  • RAII(资源获取即初始化)机制,当线程对象被销毁时,会自动执行析构函数,如果线程仍然在运行,会自动调用std::terminate()来终止程序;
  • 使用std::mutexstd::condition_variable进行同步:C++11线程库引入了互斥量(std::mutex)和条件变量(std::condition_variable),用于线程间的同步和通信。
  • 可移植性:C++11线程库提供了跨平台的线程抽象,使得编写可移植的多线程代码更加容易。开发者不再需要处理底层的线程创建和管理细节,而是直接使用标准线程库提供的接口。

1. 互斥

1.1 锁是实现互斥的方法,在std中实现了多种基本锁如下:

  1. std::mutex:最基本的互斥锁,只能在同一线程中进行加锁和解锁操作。

  2. std::recursive_mutex:递归互斥锁,允许同一线程多次加锁,但必须在同一线程中解锁相同次数。

  3. std::timed_mutex:定时互斥锁,允许线程尝试加锁一段时间,如果超时则放弃锁。

  4. std::recursive_timed_mutex:递归定时互斥锁,结合了递归和定时互斥锁的特性。

  5. std::shared_mutex:共享互斥锁,允许多个线程同时读取共享资源,但只允许一个线程写入共享资源。当一个线程持有写入权限时,其他所有线程都无法获取读取或写入权限,直到该线程释放写入权限。注意这是一个读写分离锁,即多个线程可以同时读取共享数据,但只有一个线程可以写入共享数据。当一个线程正在写入共享数据时,其他线程必须等待该线程完成写入操作后才能进行读取或写入操作。

  6. std::shared_timed_mutex:定时共享互斥锁,结合了共享和定时互斥锁的特性。

    这些锁类型都是线程安全的,可以在多线程环境下使用。互斥量的锁定和解锁操作是原子的,即它们在并发环境中是不可分割的操作,确保了线程之间的同步和互斥。这意味着在任何给定时间点,只有一个线程可以持有互斥量的锁,从而保证了共享资源的独占访问。

1.2 锁的封装

为了更方便的管理互斥锁的生命周期,具有RAII特性,即在对象构造时获取资源,在对象析构时释放资源,从而保证资源的正确获取和释放。他利用此原理在析构函数中释放了锁,即可以达到自动释放锁的效果。

  • std::unique_lock<std::mutex> lock(mutex_)

    可以在构造函数中传入一个互斥锁对象,当std::unique_lock对象被销毁时(出了作用域会自动销毁),它会自动释放互斥锁。同时,std::unique_lock还提供了lock()unlock()方法,可以手动控制互斥锁的加锁和解锁。

    /*unique_lock简单实现*/
    template<class Mutex>
    class unique_lock {
    public:
    explicit unique_lock(Mutex& m) : mutex_(m) {
    mutex_.lock();
    }
    ~unique_lock() {
    mutex_.unlock();
    }
    private:
    Mutex& mutex_;
    };
  • std::lock_guard<std::mutex> lock(mutex_).当std::lock_guard对象被销毁时,互斥锁会自动解锁。这样可以确保在std::lock_guard对象的生命周期内,互斥锁一直处于锁定状态,从而避免了忘记解锁的问题。

  • std::scoped_lock也是一个模板类,用于管理多个互斥锁。在创建std::scoped_lock对象时,需要传入多个互斥锁对象,该对象会在std::scoped_lock的构造函数中被锁定,当std::scoped_lock对象被销毁时,所有互斥锁会自动解锁。这样可以确保在std::scoped_lock对象的生命周期内,所有互斥锁都处于锁定状态,从而避免了死锁的问题。需要注意的是,std::scoped_lock可以同时锁定多个互斥锁,但是不能锁定同一个互斥锁多次,否则会导致死锁。

2.同步机制

在linux中,可以用于实现同步的机制有条件变量,信号量,互斥锁,在C++11中的同步机制主要是通过条件变量来实现的,需要引入头文件condition_variable,其主要方法有

  • wait(lock, predicate):lambda函数重载等待条件
  • notify_one():至少唤醒一个线程
  • notify_all():唤醒所有线程(如果可用资源不只有1个,竞争互斥锁失败的线程重新等待)
void WorkerThread()
{
std::cout << "WorkerThread: Started" << std::endl; // 等待主线程发出信号
std::unique_lock<std::mutex> lock(mtx);
cv.wait(lock, [] { return isReady; }); // 收到信号后执行任务
std::cout << "WorkerThread: Resumed" << std::endl;
std::cout << "WorkerThread: Performing task..." << std::endl;
// 执行任务的代码 std::cout << "WorkerThread: Completed" << std::endl;
}
int main()
{
// 创建工作线程
std::thread worker(WorkerThread);
// 发送信号给工作线程
{
std::lock_guard<std::mutex> lock(mtx);
isReady = true;
cv.notify_one();
}
// 等待工作线程完成
worker.join();
return 0;
}

可以使用条件变量与互斥锁实现信号量:

#include <mutex>
#include <condition_variable> class Semaphore {
public:
Semaphore(int count = 0) : count_(count) {} void Wait() {
std::unique_lock<std::mutex> lock(mutex_);
while (count_ == 0) {
condition_.wait(lock);
}
count_--;
} void Signal() {
std::unique_lock<std::mutex> lock(mutex_);
count_++;
condition_.notify_one();
} private:
int count_;
std::mutex mutex_;
std::condition_variable condition_;
};

3. 线程池

#include <iostream>
#include <thread>
#include <vector>
#include <queue>
#include <functional>
#include <mutex>
#include <condition_variable> class ThreadPool {
public:
ThreadPool(size_t numThreads) : stop(false) {
for (size_t i = 0; i < numThreads; ++i) {
threads.emplace_back([this]() {
while (true) {
std::function<void()> task;
{
std::unique_lock<std::mutex> lock(queueMutex);
condition.wait(lock, [this]() {
return stop || !tasks.empty();
});
if (stop && tasks.empty()) {
return;
}
task = std::move(tasks.front());
tasks.pop();
}
task();
}
});
}
} ~ThreadPool() {
{
std::unique_lock<std::mutex> lock(queueMutex);
stop = true;
}
condition.notify_all();
for (std::thread& thread : threads) {
thread.join();
}
} template<class F, class... Args>
void enqueue(F&& f, Args&&... args) {
{
std::unique_lock<std::mutex> lock(queueMutex);
tasks.emplace([=]() {
std::invoke(std::forward<F>(f), std::forward<Args>(args)...);
});
}
condition.notify_one();
} private:
std::vector<std::thread> threads;
std::queue<std::function<void()>> tasks;
std::mutex queueMutex;
std::condition_variable condition;
bool stop;
}; // 示例任务函数
void taskFunction(int id) {
std::cout << "Task " << id << " is running." << std::endl;
std::this_thread::sleep_for(std::chrono::seconds(1));
std::cout << "Task " << id << " is done." << std::endl;
} int main() {
ThreadPool threadPool(4); // 添加任务到线程池
for (int i = 0; i < 8; ++i) {
threadPool.enqueue(taskFunction, i);
} // 等待任务执行完毕
std::this_thread::sleep_for(std::chrono::seconds(5)); return 0;
}

这个地方可能会有一些拓展问题:

  • 如何安排线程优先级?
  • 线程异常时该怎么做?

C++11 同步与互斥的更多相关文章

  1. C++ 11 线程的同步与互斥

    这次写的线程的同步与互斥,不依赖于任何系统,完全使用了C++11标准的新特性来写的,就连线程函数都用了C++11标准的lambda表达式. /* * thread_test.cpp * * Copyr ...

  2. 转载自~浮云比翼:Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥)

    Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥)   介绍:什么是线程,线程的优点是什么 线程在Unix系统下,通常被称为轻量级的进程,线程虽然不是进程,但却可 ...

  3. Step by Step:Linux C多线程编程入门(基本API及多线程的同步与互斥)

    介绍:什么是线程,线程的优点是什么 线程在Unix系统下,通常被称为轻量级的进程,线程虽然不是进程,但却可以看作是Unix进程的表亲,同一进程中的多条线程将共享该进程中的全部系统资源,如虚拟地址空间, ...

  4. exec函数族,守护进程,线程同步和互斥

    2015.3.2 进程和程序有三点不同:1,存在位置不同,程序:硬盘,磁盘.进程:内存2. 程序是静态的,进程是动态的 执行./a.out -->bash->bash程序调用fork()- ...

  5. Windows下C++多线程同步与互斥简单运用

    1.  互斥量,Mutex #include <Windows.h> #include <iostream> using namespace std; DWORD WINAPI ...

  6. UNIX环境高级编程——线程同步之互斥量

    互斥量(也称为互斥锁)出自POSIX线程标准,可以用来同步同一进程中的各个线程.当然如果一个互斥量存放在多个进程共享的某个内存区中,那么还可以通过互斥量来进行进程间的同步. 互斥量,从字面上就可以知道 ...

  7. java同步和互斥【用具体程序说明】

    java同步和互斥[用具体程序说明]            所有对象都自动含有单一的锁,也就是所有对象都有且只有唯一的锁,所以当某个任务(线程)访问一个类A中含有sycnhronized的方法是,那么 ...

  8. Linux驱动之同步、互斥、阻塞的应用

    同步.互斥.阻塞的概念: 同步:在并发程序设计中,各进程对公共变量的访问必须加以制约,这种制约称为同步. 互斥机制:访问共享资源的代码区叫做临界区,这里的共享资源可能被多个线程需要,但这些共享资源又不 ...

  9. c++11 线程的互斥量

    c++11 线程的互斥量 为什么需要互斥量 在多任务操作系统中,同时运行的多个任务可能都需要使用同一种资源.这个过程有点类似于,公司部门里,我在使用着打印机打印东西的同时(还没有打印完),别人刚好也在 ...

  10. python-Lock锁线程同步和互斥

    #!/usr/bin/python #coding=utf-8 #线程间通信的同步与互斥操作-锁 import threading a=b=0 lock=threading.Lock() def va ...

随机推荐

  1. 在Linux服务器上装jenkins(方式一:war包)

    官网下载jenkins https://www.jenkins.io/zh/download/ 官网下载太慢,可以去清华镜像下载(百度搜索:清华镜像) 下载的文件是jenkins.war 把安装文件放 ...

  2. 一键式调试工具—Reqable 使用指南

    简介 Reqable是一款跨平台的专业HTTP开发和调试工具,在全平台支持HTTP1.HTTP2和HTTP3(QUIC)协议,简单易用.功能强大.性能高效,助力程序开发和测试人员提高生产力!本产品需要 ...

  3. vulntarget-a-wp

    vulntarget-a 信息收集 存活扫描,目标开放了445还是win7,考虑一手永恒之蓝 arp-scan -l nmap -A -sT -sV 192.168.130.4 永恒之蓝 用nmap的 ...

  4. 部署堡垒机4——CentOS7 编译安装 Python 3.8.12

    1.去python3的官方网站下载源代码 https://www.python.org/downloads/ 下载安装Python 3.8.12到/opt/python3 cd /opt wget h ...

  5. websocket实现实时直播

    websocket实现实时直播 这篇文章我首发于简书,拿到这里发表不过分吧?点个赞再走呗! 作为一名web开发者,我使用websocket实现实时直播(滑鸡版). 为什么是滑鸡版呢?因为他上不了生产, ...

  6. MyBatis 源码解析

    本文源码解析针对的是 MyBatis 3.4.4 MyBatis 执行流程 第一阶段 MyBatis 在这个阶段获得 Mapper 的动态代理对象,具体逻辑如下图所示: 其中,Configuratio ...

  7. 2023-08-26:请用go语言编写。开心一下的智力题: 有一个村庄,一共250人, 每一个村民要么一定说谎,要么只说真话, 村里有A、B、C、D四个球队,且每个村民只会喜欢其中的一支球队, 但是说

    2023-08-26:请用go语言编写.开心一下的智力题: 有一个村庄,一共250人, 每一个村民要么一定说谎,要么只说真话, 村里有A.B.C.D四个球队,且每个村民只会喜欢其中的一支球队, 但是说 ...

  8. 记录:excel导入导出js-xlsx,处理合并

    效果 前情提要 后端传excel坐标数据,前端自己处理模板,找资料后,选择直接载入xlsx方式. 准备工作 npm i xlsx import * as XLSX from 'xlsx' 方法一:数据 ...

  9. 神经网络基础篇:梯度下降法(Gradient Descent)

    梯度下降法 梯度下降法可以做什么? 在 测试集上,通过最小化代价函数(成本函数)\(J(w,b)\)来训练的参数\(w\)和\(b\), 如图,在第二行给出和之前一样的逻辑回归算法的代价函数(成本函数 ...

  10. 干货分享丨从MPG 线程模型,探讨Go语言的并发程序

    摘要:Go 语言的并发特性是其一大亮点,今天我们来带着大家一起看看如何使用 Go 更好地开发并发程序. 我们都知道计算机的核心为 CPU,它是计算机的运算和控制核心,承载了所有的计算任务.最近半个世纪 ...