0

0

c++中如何实现一个简单的信号量Semaphore_c++多线程同步【详解】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-16 18:02:02

|

201人浏览过

|

来源于php中文网

原创

C++17及更早版本无原生信号量,需用std::condition_variable+std::mutex+计数器手动实现;C++20引入std::counting_semaphore,但依赖编译器和系统支持,手写轻量Semaphore类更稳妥,关键为m_count、m_mutex、m_cv三要素。

c++中如何实现一个简单的信号量semaphore_c++多线程同步【详解】

为什么标准 C++ 没有 std::semaphore(直到 C++20)

在 C++17 及更早版本中,std::mutexstd::condition_variable 等是同步主力,但没有原生信号量。很多开发者误以为缺了它就写不了资源计数控制——其实可以用 std::condition_variable + std::mutex + 计数器手动模拟,而且逻辑清晰、可控性强。

C++20 引入了 std::counting_semaphore(注意不是 std::semaphore),但它要求编译器支持(如 GCC 11+、Clang 12+、MSVC 19.29+),且底层依赖系统信号量或 futex,跨平台行为未必一致。如果你的目标环境不明确支持 C++20,手写一个轻量 Semaphore 类反而更稳妥。

手写 Semaphore 类的关键成员和线程安全设计

核心是三个东西:一个整型计数器(m_count)、一把互斥锁(m_mutex)、一个条件变量(m_cv)。acquire() 阻塞直到计数 > 0;release() 增加计数并唤醒等待者。

常见错误包括:未用 while 循环检查条件(虚假唤醒)、release() 不加锁(导致 m_count 竞态)、忘记在 acquire() 中用 wait() 的 predicate 版本。

立即学习C++免费学习笔记(深入)”;

  • acquire() 必须用 m_cv.wait(m_mutex, [&]{ return m_count > 0; });,不能用 if (m_count
  • release() 要先加锁、改计数、再 notify_one()notify_all();若只唤醒一个,多个等待线程可能仍阻塞
  • 构造时传入初始值,应校验非负:若为负,抛 std::invalid_argument

C++11 兼容的手写 Semaphore 示例

以下实现无外部依赖,可直接编译运行(C++11 起):

Type Studio
Type Studio

一个视频编辑器,提供自动转录、自动生成字幕、视频翻译等功能

下载
#include 
#include 

class Semaphore {
    std::mutex m_mutex;
    std::condition_variable m_cv;
    int m_count;

public:
    explicit Semaphore(int count = 0) : m_count(count) {
        if (count < 0) throw std::invalid_argument("Semaphore count cannot be negative");
    }

    void acquire() {
        std::unique_lock lock(m_mutex);
        m_cv.wait(lock, [&] { return m_count > 0; });
        --m_count;
    }

    void release() {
        std::unique_lock lock(m_mutex);
        ++m_count;
        m_cv.notify_one(); // 或 notify_all(),依场景选
    }
};

使用示例:限制最多 2 个线程同时访问某资源:

#include 
#include 

Semaphore sem(2);

void worker(int id) {
    sem.acquire();
    std::cout << "Worker " << id << " entered\n";
    std::this_thread::sleep_for(std::chrono::seconds(1));
    std::cout << "Worker " << id << " leaving\n";
    sem.release();
}

int main() {
    std::thread t1(worker, 1);
    std::thread t2(worker, 2);
    std::thread t3(worker, 3);
    std::thread t4(worker, 4);
    t1.join(); t2.join(); t3.join(); t4.join();
}

std::counting_semaphore(C++20)要注意什么

虽然标准提供了,但实际使用时容易忽略两点:

  • 它不提供 try_acquire() 的超时版本(C++20 只有 try_acquire() 无参版),需要配合 std::chrono 手动轮询,性能差
  • 某些平台(如 Windows MinGW)可能未完全实现,std::counting_semaphore 行为可能退化为互斥锁,失去计数语义
  • 初始化值必须是常量表达式(constexpr),不能是运行时变量,否则编译失败

所以即使你用 C++20,对简单场景(如限流、池管理),手写 Semaphore 依然更透明、易调试、兼容性更好。

真正复杂的是需要跨进程、带优先级唤醒、或与操作系统信号量深度绑定的场景——那得用 POSIX sem_t 或 Windows CreateSemaphore,而不是标准库。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

740

2023.08.22

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

88

2023.09.25

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

480

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

143

2025.12.24

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

2

2026.01.16

全民K歌得高分教程大全
全民K歌得高分教程大全

本专题整合了全民K歌得高分技巧汇总,阅读专题下面的文章了解更多详细内容。

0

2026.01.16

C++ 单元测试与代码质量保障
C++ 单元测试与代码质量保障

本专题系统讲解 C++ 在单元测试与代码质量保障方面的实战方法,包括测试驱动开发理念、Google Test/Google Mock 的使用、测试用例设计、边界条件验证、持续集成中的自动化测试流程,以及常见代码质量问题的发现与修复。通过工程化示例,帮助开发者建立 可测试、可维护、高质量的 C++ 项目体系。

10

2026.01.16

java数据库连接教程大全
java数据库连接教程大全

本专题整合了java数据库连接相关教程,阅读专题下面的文章了解更多详细内容。

33

2026.01.15

Java音频处理教程汇总
Java音频处理教程汇总

本专题整合了java音频处理教程大全,阅读专题下面的文章了解更多详细内容。

15

2026.01.15

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号