0

0

如何在c++中实现一个简单的Futures/Promises库? (异步编程基础)

冰火之心

冰火之心

发布时间:2026-01-10 12:02:02

|

141人浏览过

|

来源于php中文网

原创

自己实现最小Promise/Future是为了透彻理解值跨线程传递、状态原子切换、回调注册触发三大核心机制,避开标准库复杂调度细节。

如何在c++中实现一个简单的futures/promises库? (异步编程基础)

为什么 std::promise/std::future 已经够用,还要自己实现?

因为标准库std::promisestd::future 是线程安全、带状态机、支持 wait_for/wait_until 的完整实现,但它的内部依赖 std::mutexstd::condition_variable 甚至 TLS(如 std::async 的 launch policy),初学者直接读源码容易迷失在调度细节里。自己写一个最小可行版本,能看清「值如何跨线程传递」「状态如何原子切换」「回调如何注册与触发」这三个核心问题。

最简 Promise/Future 需要哪些成员?

一个可拷贝的 Future + 一个独占持有的 Promise 就够了,不支持共享状态(即不实现 std::shared_future),也不支持多消费者。关键点是:状态必须用原子变量控制,值存储需延迟构造(避免未就绪时访问)。

  • Promise 持有 std::atomic 表示是否已设置(is_ready_),以及一个 std::optional 存值(C++17 起可用;否则用 aligned_storage + placement new)
  • Future 持有对同一份 is_ready_value_ 的引用(或指针),只读,不可调用 set_value
  • 禁止拷贝 Promise,只允许移动 —— 否则多个对象可能同时调用 set_value
  • Future::get() 必须阻塞直到 is_ready_ == true,可用 while(!is_ready_.load()) std::this_thread::yield();(简单场景),生产环境应改用条件变量
#include 
#include 
#include 
#include 

template class SimplePromise { std::atomic_bool isready{false}; std::optional value_;

public: SimplePromise() = default; SimplePromise(const SimplePromise&) = delete; SimplePromise& operator=(const SimplePromise&) = delete; SimplePromise(SimplePromise&&) = default; SimplePromise& operator=(SimplePromise&&) = default;

void set_value(T v) {
    assert(!is_ready_.exchange(true)); // 确保只设置一次
    value_.emplace(std::move(v));
}

friend class SimpleFuturezuojiankuohaophpcnTyoujiankuohaophpcn;

};

Morph Studio
Morph Studio

Morph Studio是一款领先的文字转视频AI平台,可以将用户输入的文字转化为精美视频。

下载

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

template class SimpleFuture { SimplePromise* promise_;

public: explicit SimpleFuture(SimplePromise& p) : promise_(&p) {}

T get() {
    while (!promise_-youjiankuohaophpcnis_ready_.load()) {
        std::this_thread::yield();
    }
    return std::move(*promise_-youjiankuohaophpcnvalue_);
}

};

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

如何添加回调支持(类似 then())?

原生 std::future 不支持链式回调,这是自己实现时最有价值的扩展点。核心是:当 Promise 就绪时,遍历并执行所有注册的函数对象(std::function)。注意两点:回调必须在就绪后立即执行(不新开线程),且不能递归调用 set_value(否则溢出)。

  • std::vector<:function>> 加进 SimplePromise,并在 set_value 末尾遍历调用
  • Future::then() 返回一个新的 Future,它内部新建一个 Promise,并在回调里调用其 set_value
  • 若回调抛异常,需捕获并转为 std::exception_ptr(否则程序终止)
  • 不要在回调里直接访问原 Futureget() —— 此时值已就绪,但语义上应通过参数传入

常见崩溃点和调试建议

自己实现最容易栽在内存生命周期和竞态上:

  • Future 持有裸指针指向栈上的 Promise?一旦 Promise 析构,Future::get() 就访问野指针 —— 必须确保 Promise 生命周期长于所有关联的 Future
  • 多个线程同时调用同一个 Promise::set_value()is_ready_.exchange(true) 是原子的,但 value_.emplace() 不是 —— 所以 assert 那行其实承担了排他性检查职责,漏掉就 UB
  • std::this_thread::yield() 做轮询?在单核机器或高负载下可能饿死;真实项目中应搭配 std::condition_variablestd::atomic_flag::wait()(C++20)
  • 移动 Promise 后又调用 set_value?移动后原对象处于有效但未指定状态,is_ready_.load() 仍合法,但 value_.emplace() 可能失败 —— 所以移动构造函数里应显式置空 value_ 并重置 is_ready_

真正难的不是语法,而是想清楚:谁拥有值、谁负责析构、谁决定就绪时机、回调在哪个线程执行。这些边界模糊的地方,才是异步逻辑出 bug 的温床。

相关专题

更多
while的用法
while的用法

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

83

2023.09.25

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

175

2023.11.23

java中void的含义
java中void的含义

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

95

2025.11.27

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

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

384

2023.07.18

堆和栈区别
堆和栈区别

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

568

2023.08.10

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

463

2024.01.03

python中class的含义
python中class的含义

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

12

2025.12.06

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

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

479

2023.08.10

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

25

2026.01.09

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

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

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