0

0

C++中的std::barrier是什么?(如何循环使用同步点)

尼克

尼克

发布时间:2026-03-03 14:44:03

|

595人浏览过

|

来源于php中文网

原创

std::barrier 默认不可循环使用,因其 expected 阈值为一次性;需用带回调函数的构造函数(如 std::barrier b(n, [](std::ptrdiff_t){}))才能支持重复同步。

c++中的std::barrier是什么?(如何循环使用同步点)

std::barrier 为什么不能直接循环用

它不是设计成“用完还能接着用”的同步原语。构造时指定的到达次数 expected 是一次性阈值,所有线程到达后,std::barrier 进入“释放态”,后续调用 arrive()wait() 会触发未定义行为(常见表现是卡死、崩溃或 std::terminate)。这不是 bug,是标准明确规定的生命周期限制。

常见错误现象:

  • 循环里反复调用同一个 barrier.wait(),第二次就 hang 住
  • barrier 声明在循环外,期望复用——实际不可行
  • 没注意到 barrier 析构后再次访问成员函数,UB 难调试

怎么实现“可循环的同步点”

标准库不提供内置循环版 barrier,但有可靠替代方案:用 std::latch + 重置逻辑,或更推荐直接用 std::barrier 的“可重用变体”——即 C++20 引入的 std::barrier 构造函数重载,支持传入回调函数,且对象本身支持重复使用(前提是用对构造方式)。

关键点:

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

  • 必须用带回调的构造函数:std::barrier<int> b(n, [](int) {})</int>,其中回调参数类型要匹配(C++20 要求是 std::ptrdiff_t 或能隐式转为它的类型)
  • 这个版本的 barrier 在每次所有线程到达后自动重置计数器,无需重建对象
  • 不要用无参构造或只传 expected 的构造函数,那仍是单次版

示例:

Qwen
Qwen

阿里巴巴推出的一系列AI大语言模型和多模态模型

下载
std::barrier b(4, [](std::ptrdiff_t) {}); // ✅ 可循环
for (int i = 0; i < 3; ++i) {
    std::jthread t([&b](int id) {
        // 工作...
        b.arrive_and_wait(); // 每次都有效
    }, i);
}

std::barrier 和 std::latch 的核心区别在哪

别混用场景。两者语义不同,选错会导致逻辑错或性能浪费。

  • std::latch 是“一次性门闩”:只能 count_down() 到零一次,之后 wait() 总是立即返回;无法重置,也不能循环等同一批线程多次
  • std::barrier(带回调的)是“可重置同步点”:每次全员到达后自动归零并准备下一轮,适合循环任务分片(如多轮并行计算)
  • 性能上,barrier 通常比手动管理多个 latch 更轻量,因为内核态等待更少、状态更紧凑

典型误用:用 std::latch 替代循环 barrier,结果每轮都得 new 一个新 latch,堆分配开销大,还容易漏 delete 或悬空引用。

容易被忽略的线程安全与生命周期细节

std::barrier 对象本身是线程安全的,但它的生命周期必须严格覆盖所有调用它的线程。这是最常踩的坑。

  • 如果 barrier 是局部变量,而线程是 std::jthread 或 detach 的,函数返回后 barrier 析构,线程还在跑 —— UB
  • 回调函数里不要捕获或操作已销毁的资源(比如回调中调用 delete this 或访问栈变量)
  • arrive_and_wait() 返回前,所有线程都还没离开屏障点,此时修改 barrier 状态(比如重新赋值)是未定义的

稳妥做法:把 std::barrier 放在作用域足够长的地方,比如类成员、全局 static,或用 std::shared_ptr 管理其生命周期。

复杂点在于,它不像 mutex 那样有明显“加锁失败”的错误提示;出问题往往静默卡死或随机崩溃,调试时得盯住对象生存期和调用顺序。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

910

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

598

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

294

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

210

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

432

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

600

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

432

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

600

2023.08.10

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

0

2026.03.03

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 10.5万人学习

C 教程
C 教程

共75课时 | 5.1万人学习

C++教程
C++教程

共115课时 | 20.2万人学习

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

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