首页 > 后端开发 > C++ > 正文

c++如何使用std::future和std::promise_c++异步编程future/promise指南

尼克
发布: 2025-09-18 10:50:01
原创
934人浏览过
std::future和std::promise用于C++异步编程,前者获取结果,后者设置结果;通过创建promise、获取future、启动线程、设置值或异常、最后get获取结果实现;get阻塞可用wait_for避免;异常通过set_exception传递;shared_future允许多次get;async封装了future/promise简化使用;需注意数据线程安全及promise生命周期。

c++如何使用std::future和std::promise_c++异步编程future/promise指南

C++ 中

std::future
登录后复制
std::promise
登录后复制
是实现异步编程的关键工具。它们允许你在不同的线程之间传递数据和状态,而无需显式地使用锁或条件变量。简单来说,
std::promise
登录后复制
负责设置异步操作的结果,而
std::future
登录后复制
负责获取这个结果。

解决方案

要使用

std::future
登录后复制
std::promise
登录后复制
进行异步编程,你需要遵循以下步骤:

  1. 创建

    std::promise
    登录后复制
    对象:
    std::promise
    登录后复制
    对象用于设置异步操作的结果。你需要指定结果的类型。例如,如果你要异步计算一个整数,可以创建一个
    std::promise<int>
    登录后复制
    对象。

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

  2. 获取

    std::future
    登录后复制
    对象:
    std::promise
    登录后复制
    对象中获取一个
    std::future
    登录后复制
    对象。
    std::future
    登录后复制
    对象允许你稍后获取异步操作的结果。

  3. 启动异步任务 将异步任务提交给一个新线程或线程池。在这个任务中,你需要使用

    std::promise
    登录后复制
    对象来设置结果。

  4. 设置结果: 在异步任务中,使用

    std::promise::set_value()
    登录后复制
    方法设置结果。如果异步任务抛出异常,可以使用
    std::promise::set_exception()
    登录后复制
    方法设置异常。

  5. 获取结果: 在主线程中,使用

    std::future::get()
    登录后复制
    方法获取异步操作的结果。
    get()
    登录后复制
    方法会阻塞,直到结果可用。如果异步任务抛出了异常,
    get()
    登录后复制
    方法会重新抛出该异常。

下面是一个简单的示例:

#include <iostream>
#include <future>
#include <thread>

int calculate_sum(int a, int b) {
    // 模拟耗时操作
    std::this_thread::sleep_for(std::chrono::seconds(2));
    return a + b;
}

int main() {
    std::promise<int> promise;
    std::future<int> future = promise.get_future();

    std::thread t([&promise]() {
        try {
            int result = calculate_sum(5, 3);
            promise.set_value(result);
        } catch (...) {
            promise.set_exception(std::current_exception());
        }
    });

    try {
        int sum = future.get();
        std::cout << "Sum: " << sum << std::endl;
    } catch (const std::exception& e) {
        std::cerr << "Exception: " << e.what() << std::endl;
    }

    t.join();

    return 0;
}
登录后复制

副标题1

std::future
登录后复制
get()
登录后复制
方法会阻塞,如何避免阻塞?

可以使用

std::future::wait_for()
登录后复制
std::future::wait_until()
登录后复制
方法来避免无限期阻塞。这些方法允许你指定一个超时时间。如果在超时时间内结果不可用,它们会返回一个
std::future_status
登录后复制
值,指示超时或结果已准备好。

#include <iostream>
#include <future>
#include <chrono>
#include <thread>

int main() {
    std::promise<int> promise;
    std::future<int> future = promise.get_future();

    std::thread t([&promise]() {
        std::this_thread::sleep_for(std::chrono::seconds(3));
        promise.set_value(42);
    });

    // 等待最多2秒
    auto status = future.wait_for(std::chrono::seconds(2));

    if (status == std::future_status::ready) {
        std::cout << "Result: " << future.get() << std::endl;
    } else if (status == std::future_status::timeout) {
        std::cout << "Timeout: Result not available yet." << std::endl;
    } else {
        std::cout << "Deferred." << std::endl;
    }

    t.join();
    return 0;
}
登录后复制

副标题2

豆包AI编程
豆包AI编程

豆包推出的AI编程助手

豆包AI编程 483
查看详情 豆包AI编程

std::promise
登录后复制
std::future
登录后复制
如何处理异常?

std::promise
登录后复制
允许你使用
set_exception()
登录后复制
方法设置一个异常。
std::future
登录后复制
get()
登录后复制
方法会重新抛出这个异常。这允许你将异步操作中的异常传递回主线程。
std::current_exception()
登录后复制
可以捕获当前线程的异常并将其存储起来,以便稍后通过
std::promise
登录后复制
传递。

#include <iostream>
#include <future>
#include <thread>
#include <stdexcept>

int main() {
    std::promise<int> promise;
    std::future<int> future = promise.get_future();

    std::thread t([&promise]() {
        try {
            throw std::runtime_error("Something went wrong in the thread!");
        } catch (...) {
            promise.set_exception(std::current_exception());
        }
    });

    try {
        future.get(); // 这会抛出 std::runtime_error
    } catch (const std::runtime_error& e) {
        std::cerr << "Caught exception: " << e.what() << std::endl;
    }

    t.join();
    return 0;
}
登录后复制

副标题3

std::shared_future
登录后复制
std::future
登录后复制
有什么区别?什么时候应该使用
std::shared_future
登录后复制

std::future
登录后复制
只能被
get()
登录后复制
调用一次,之后就失效了。
std::shared_future
登录后复制
可以被多个线程共享,并且每个线程都可以多次调用
get()
登录后复制
(或者
wait()
登录后复制
等)。

如果你需要多个线程访问同一个异步操作的结果,应该使用

std::shared_future
登录后复制
std::shared_future
登录后复制
允许你创建多个对同一个结果的引用。

#include <iostream>
#include <future>
#include <thread>
#include <vector>

int main() {
    std::promise<int> promise;
    std::future<int> future = promise.get_future();
    std::shared_future<int> shared_future = future.share();

    std::vector<std::thread> threads;
    for (int i = 0; i < 3; ++i) {
        threads.emplace_back([shared_future, i]() {
            std::cout << "Thread " << i << ": " << shared_future.get() << std::endl;
        });
    }

    promise.set_value(123);

    for (auto& t : threads) {
        t.join();
    }

    return 0;
}
登录后复制

副标题4

std::async
登录后复制
std::future/std::promise
登录后复制
的关系是什么?

std::async
登录后复制
是一个高层次的异步操作启动函数,它内部使用了
std::future
登录后复制
std::promise
登录后复制
。当你调用
std::async
登录后复制
时,它会自动创建一个
std::promise
登录后复制
,启动一个异步任务,并将
std::promise
登录后复制
关联的
std::future
登录后复制
返回给你。
std::async
登录后复制
简化了异步编程,你不需要手动创建
std::promise
登录后复制
std::thread
登录后复制
。但是,如果你需要更精细的控制(例如,自定义线程池或异常处理),那么手动使用
std::future
登录后复制
std::promise
登录后复制
可能会更好。

#include <iostream>
#include <future>

int calculate_product(int a, int b) {
    std::cout << "Calculating product in a separate thread." << std::endl;
    return a * b;
}

int main() {
    std::future<int> future = std::async(std::launch::async, calculate_product, 7, 6);

    std::cout << "Waiting for the result..." << std::endl;
    int product = future.get();
    std::cout << "Product: " << product << std::endl;

    return 0;
}
登录后复制

副标题5

使用

std::future
登录后复制
std::promise
登录后复制
时需要注意哪些线程安全问题?

std::promise
登录后复制
std::future
登录后复制
本身是线程安全的,但你需要注意它们所操作的数据的线程安全性。例如,如果你在异步任务中修改一个共享变量,你需要使用锁来保护这个变量。另外,确保在
std::promise
登录后复制
对象销毁之前设置结果或异常,否则会导致未定义行为。 避免在多个线程中同时调用同一个
std::promise
登录后复制
对象的
set_value
登录后复制
set_exception
登录后复制
方法。

以上就是c++++如何使用std::future和std::promise_c++异步编程future/promise指南的详细内容,更多请关注php中文网其它相关文章!

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

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