0

0

怎样在C++协程中处理异常 协程帧中的异常捕获与传播

P粉602998670

P粉602998670

发布时间:2025-07-14 09:56:02

|

838人浏览过

|

来源于php中文网

原创

c++++协程中处理异常的关键在于利用try...catch块捕获异常、处理co_await表达式抛出的异常,并通过promise对象的unhandled_exception()方法管理未处理异常。1. 在协程体内直接使用try...catch捕获并处理异常;2. co_await表达式若引发异常,会传播至协程体,需在try...catch中捕获;3. 未处理的异常会调用promise的unhandled_exception()方法,默认终止程序,可自定义保存异常;4. 异常可在协程链中通过set_exception传递;5. 使用raii或智能指针避免异常导致资源泄漏。

怎样在C++协程中处理异常 协程帧中的异常捕获与传播

在C++协程中处理异常,本质上就是在协程帧的生命周期内,确保即使发生错误,程序也能优雅地恢复或终止,避免崩溃。 这需要我们理解协程的挂起和恢复机制,并巧妙地利用try...catch块。

怎样在C++协程中处理异常 协程帧中的异常捕获与传播

协程帧中的异常捕获与传播,主要依赖于co_await操作符和promise对象。 当协程体内部抛出异常时,这个异常需要能够被“传递”到调用者或者协程的promise对象,以便进行处理。

处理协程中的异常,需要关注几个关键点:一是协程体内的异常捕获,二是co_await表达式中的异常处理,三是promise对象的异常处理机制。

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

怎样在C++协程中处理异常 协程帧中的异常捕获与传播

如何在协程体内捕获异常?

最直接的方式,就是在协程函数体内使用try...catch块。 这和普通的C++函数异常处理没有本质区别,但需要注意的是,协程可能会挂起,因此异常发生的位置和处理的位置可能不在同一个函数调用栈上。

例如:

怎样在C++协程中处理异常 协程帧中的异常捕获与传播
#include 
#include 

struct MyCoroutine {
    struct promise_type {
        int result;

        MyCoroutine get_return_object() { return {std::coroutine_handle::from_promise(*this)}; }
        std::suspend_never initial_suspend() { return {}; }
        std::suspend_never final_suspend() noexcept { return {}; }
        void unhandled_exception() { std::terminate(); }
        void return_value(int value) { result = value; }
    };

    std::coroutine_handle handle;
};


MyCoroutine my_coroutine() {
    try {
        // 模拟可能抛出异常的操作
        throw std::runtime_error("Something went wrong!");
        co_return 42; // 如果没有异常,返回42
    } catch (const std::exception& e) {
        std::cerr << "Caught exception in coroutine: " << e.what() << std::endl;
        co_return -1; // 发生异常时,返回-1
    }
}

int main() {
    MyCoroutine coro = my_coroutine();
    std::cout << "Result: " << coro.handle.promise().result << std::endl;
    coro.handle.destroy();
    return 0;
}

在这个例子中,如果throw std::runtime_error被执行,那么异常会被catch块捕获,并打印错误信息。 协程最终会co_return -1,将-1作为结果返回。

co_await 表达式如何处理异常?

co_await一个可能抛出异常的awaitable对象时,异常的处理会稍微复杂一些。 如果awaitable对象的await_resume()方法抛出异常,那么这个异常会被传播到协程中,就像co_await表达式直接抛出异常一样。

#include 
#include 
#include 

struct ExceptionAwaitable {
    bool await_ready() { return false; }
    void await_suspend(std::coroutine_handle<> h) { h.resume(); }
    int await_resume() {
        throw std::runtime_error("Exception from await_resume!");
    }
};


struct MyCoroutine2 {
    struct promise_type {
        int result;

        MyCoroutine2 get_return_object() { return {std::coroutine_handle::from_promise(*this)}; }
        std::suspend_never initial_suspend() { return {}; }
        std::suspend_never final_suspend() noexcept { return {}; }
        void unhandled_exception() { std::terminate(); }
        void return_value(int value) { result = value; }
    };

    std::coroutine_handle handle;
};


MyCoroutine2 my_coroutine2() {
    try {
        int value = co_await ExceptionAwaitable{};
        std::cout << "Value after co_await: " << value << std::endl; // 不会被执行
        co_return 42;
    } catch (const std::exception& e) {
        std::cerr << "Caught exception after co_await: " << e.what() << std::endl;
        co_return -1;
    }
}

int main() {
    MyCoroutine2 coro = my_coroutine2();
    std::cout << "Result: " << coro.handle.promise().result << std::endl;
    coro.handle.destroy();
    return 0;
}

在这个例子中,ExceptionAwaitableawait_resume()方法会抛出一个异常。 这个异常会被my_coroutine2函数中的try...catch块捕获。

CopyWeb
CopyWeb

AI网页设计转换工具,可以将屏幕截图、网站URL转换为代码组件

下载

Promise对象的unhandled_exception()方法有什么作用?

如果协程体内的异常没有被捕获,那么会调用promise对象的unhandled_exception()方法。 默认情况下,这个方法会调用std::terminate(),导致程序终止。 我们可以自定义这个方法,来实现更灵活的异常处理。

例如,我们可以记录错误信息,或者尝试恢复协程的状态。

#include 
#include 
#include 

struct MyCoroutine3 {
    struct promise_type {
        int result;
        std::exception_ptr exception;

        MyCoroutine3 get_return_object() { return {std::coroutine_handle::from_promise(*this)}; }
        std::suspend_never initial_suspend() { return {}; }
        std::suspend_never final_suspend() noexcept { return {}; }
        void unhandled_exception() {
            exception = std::current_exception(); // 保存异常
        }
        void return_value(int value) { result = value; }
    };

    std::coroutine_handle handle;
};


MyCoroutine3 my_coroutine3() {
    throw std::runtime_error("Unhandled exception in coroutine!");
    co_return 42; // 不会被执行
}

int main() {
    MyCoroutine3 coro = my_coroutine3();

    if (coro.handle.promise().exception) {
        try {
            std::rethrow_exception(coro.handle.promise().exception);
        } catch (const std::exception& e) {
            std::cerr << "Caught unhandled exception in main: " << e.what() << std::endl;
        }
    } else {
        std::cout << "Result: " << coro.handle.promise().result << std::endl;
    }

    coro.handle.destroy();
    return 0;
}

在这个例子中,my_coroutine3函数会抛出一个未处理的异常。 这个异常会被promise对象的unhandled_exception()方法捕获,并保存到exception成员中。 在main函数中,我们可以检查exception成员是否为空,如果不为空,则重新抛出异常并进行处理。

如何在co_await链中传递异常?

在复杂的协程调用链中,异常需要在不同的协程之间传递。 这可以通过promise对象的return_void()set_exception()方法来实现。

当一个协程需要将异常传递给它的调用者时,它可以调用promise对象的set_exception()方法。 这个方法会将异常保存到promise对象中,并在协程恢复时重新抛出。

#include 
#include 
#include 

struct NestedCoroutine {
    struct promise_type {
        NestedCoroutine get_return_object() { return {std::coroutine_handle::from_promise(*this)}; }
        std::suspend_never initial_suspend() { return {}; }
        std::suspend_never final_suspend() noexcept { return {}; }
        void unhandled_exception() { std::terminate(); }
        void return_void() {} // 协程不返回值时使用
        void set_exception(std::exception_ptr e) {
            exception = e;
        }

        std::exception_ptr exception;
    };

    std::coroutine_handle handle;
};


NestedCoroutine nested_coroutine() {
    throw std::runtime_error("Exception from nested coroutine!");
    co_return; // 不会被执行
}

struct MyCoroutine4 {
    struct promise_type {
        int result;

        MyCoroutine4 get_return_object() { return {std::coroutine_handle::from_promise(*this)}; }
        std::suspend_never initial_suspend() { return {}; }
        std::suspend_never final_suspend() noexcept { return {}; }
        void unhandled_exception() { std::terminate(); }
        void return_value(int value) { result = value; }
    };

    std::coroutine_handle handle;
};


MyCoroutine4 my_coroutine4() {
    try {
        co_await nested_coroutine();
        co_return 42; // 不会被执行
    } catch (const std::exception& e) {
        std::cerr << "Caught exception in my_coroutine4: " << e.what() << std::endl;
        co_return -1;
    }
}

int main() {
    MyCoroutine4 coro = my_coroutine4();
    std::cout << "Result: " << coro.handle.promise().result << std::endl;
    coro.handle.destroy();
    return 0;
}

在这个例子中,nested_coroutine函数会抛出一个异常。 这个异常会被传递到my_coroutine4函数中,并被try...catch块捕获。

如何避免异常导致的资源泄漏?

在使用协程时,需要特别注意资源管理。 如果在协程挂起时发生异常,那么可能会导致资源泄漏。 为了避免这种情况,可以使用RAII(Resource Acquisition Is Initialization)技术,或者使用智能指针来管理资源。

另外,也可以在promise对象的final_suspend()方法中释放资源。 这个方法会在协程结束时被调用,无论协程是正常结束还是因为异常而结束。

相关专题

更多
resource是什么文件
resource是什么文件

Resource文件是一种特殊类型的文件,它通常用于存储应用程序或操作系统中的各种资源信息。它们在应用程序开发中起着关键作用,并在跨平台开发和国际化方面提供支持。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

147

2023.12.20

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

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

386

2023.07.18

堆和栈区别
堆和栈区别

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

569

2023.08.10

promise的用法
promise的用法

“promise” 是一种用于处理异步操作的编程概念,它可以用来表示一个异步操作的最终结果。Promise 对象有三种状态:pending(进行中)、fulfilled(已成功)和 rejected(已失败)。Promise的用法主要包括构造函数、实例方法(then、catch、finally)和状态转换。

298

2023.10.12

html文本框类型介绍
html文本框类型介绍

html文本框类型有单行文本框、密码文本框、数字文本框、日期文本框、时间文本框、文件上传文本框、多行文本框等等。详细介绍:1、单行文本框是最常见的文本框类型,用于接受单行文本输入,用户可以在文本框中输入任意文本,例如用户名、密码、电子邮件地址等;2、密码文本框用于接受密码输入,用户在输入密码时,文本框中的内容会被隐藏,以保护用户的隐私;3、数字文本框等等。

396

2023.10.12

Java 项目构建与依赖管理(Maven / Gradle)
Java 项目构建与依赖管理(Maven / Gradle)

本专题系统讲解 Java 项目构建与依赖管理的完整体系,重点覆盖 Maven 与 Gradle 的核心概念、项目生命周期、依赖冲突解决、多模块项目管理、构建加速与版本发布规范。通过真实项目结构示例,帮助学习者掌握 从零搭建、维护到发布 Java 工程的标准化流程,提升在实际团队开发中的工程能力与协作效率。

10

2026.01.12

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

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

106

2026.01.09

c++框架学习教程汇总
c++框架学习教程汇总

本专题整合了c++框架学习教程汇总,阅读专题下面的文章了解更多详细内容。

63

2026.01.09

学python好用的网站推荐
学python好用的网站推荐

本专题整合了python学习教程汇总,阅读专题下面的文章了解更多详细内容。

139

2026.01.09

热门下载

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

精品课程

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

共28课时 | 3万人学习

MySQL 教程
MySQL 教程

共48课时 | 1.7万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.1万人学习

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

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