新闻中心

c++怎么实现一个简单的协程库_c++手写协程实现原理与示例

2025-11-10
浏览次数:
返回列表
协程通过用户态上下文切换实现轻量级并发,本文基于C++11和ucontext库实现简易协程,展示创建、挂起与恢复机制,利用getcontext、makecontext和swapcontext完成栈隔离与执行流控制,两个协程可交替执行,体现协程核心原理。

c++怎么实现一个简单的协程库_c++手写协程实现原理与示例

协程是一种比线程更轻量的并发编程模型,能够在用户态实现函数的暂停与恢复。C++20 引入了原生协程支持,但理解如何从零实现一个简单的协程库,有助于深入掌握其底层机制。下面介绍基于 C++11+ 的手写协程实现原理,并给出一个简易示例。

协程的基本概念

协程允许函数执行到某一点时暂停,保存当前状态,之后从中断处继续执行。它不同于线程,不依赖操作系统调度,开销小,适合高并发场景。

一个最简协程需要具备:

  • 可挂起(suspend)和恢复(resume)的能力
  • 上下文切换(context switch)机制
  • 状态管理(如是否完成、返回值等)

使用 ucontext 实现上下文切换

注意:ucontext 是 POSIX 标准的一部分,在 macOS 和 Linux 上可用,Windows 不支持。

我们可以借助 getcontextsetcontextmakecontextswapcontext 来实现协程的跳转。

核心思路是:

  • 每个协程拥有独立的栈空间和上下文(ucontext_t)
  • 主函数启动协程时,通过 makecontext 创建执行环境
  • 协程运行中可通过 swapcontext 切回主函数或其他协程

简易协程库实现示例

下面是一个极简的协程实现,包含创建、切换与恢复功能:

Yaara Yaara

使用AI生成一流的文案广告,电子邮件,网站,列表,博客,故事和更多…

Yaara 95 查看详情 Yaara
#include <iostream>
#include <ucontext.h>
#include <map>
#include <functional>
#include <cstdlib>

class SimpleCoroutine {
private:
    static std::map<int, ucontext_t> coroutines;
    static int current_id;
    static ucontext_t main_context;

    ucontext_t ctx;
    char* stack;
    int id;
    bool is_done;

public:
    SimpleCoroutine(std::function<void()> func) {
        id = ++current_id;
        stack = new char[8192]; // 8KB 栈空间
        is_done = false;

        getcontext(&ctx);
        ctx.uc_stack.ss_sp = stack;
        ctx.uc_stack.ss_size = 8192;
        ctx.uc_link = &main_context; // 协程结束回到主上下文
        makecontext(&ctx, (void(*)())lambda_wrapper, 1, this);

        coroutines[id] = ctx;
    }

    ~SimpleCoroutine() {
        delete[] stack;
    }

    static void lambda_wrapper(SimpleCoroutine* self) {
        self->is_done = false;
        self->run();
        self->is_done = true;
    }

    void run() {
        // 模拟协程体
        std::cout << "协程 " << id << " 开始执行\n";
        yield(); // 第一次让出
        std::cout << "协程 " << id << " 继续执行\n";
        yield();
        std::cout << "协程 " << id << " 执行结束\n";
    }

    void resume() {
        if (is_done) return;
        swapcontext(&main_context, &coroutines[id]);
    }

    static void yield() {
        swapcontext(&coroutines[current_id], &main_context);
    }

    static void yield(int cid) {
        swapcontext(&coroutines[current_id], &main_context);
    }
};

// 静态成员定义
std::map<int, ucontext_t> SimpleCoroutine::coroutines;
int SimpleCoroutine::current_id = 0;
ucontext_t SimpleCoroutine::main_context;

使用示例

测试两个协程交替执行:

int main() {
    getcontext(&SimpleCoroutine::main_context);

    SimpleCoroutine co1([]{}); // lambda 只是为了占位,实际逻辑在 run 中
    SimpleCoroutine co2([]{});

    std::cout << "主函数启动协程1\n";
    co1.resume();

    std::cout << "主函数启动协程2\n";
    co2.resume();

    std::cout << "主函数再次恢复协程1\n";
    co1.resume();

    std::cout << "主函数再次恢复协程2\n";
    co2.resume();

    return 0;
}

输出结果大致为:

主函数启动协程1
协程 1 开始执行
主函数启动协程2
协程 2 开始执行
主函数再次恢复协程1
协程 1 继续执行
主函数再次恢复协程2
协程 2 继续执行

原理总结

这个简易协程库的核心在于:

  • 每个协程有自己的栈和上下文,通过 ucontext 系列函数管理
  • makecontext 将函数绑定到指定上下文和栈上
  • swapcontext 实现两个上下文之间的切换
  • 协程结束后自动返回主上下文(uc_link 设置)

虽然这只是一个玩具级实现,但它展示了协程的本质:控制流的主动让出与恢复。

基本上就这些。真实生产环境中的协程库(如 Boost.Context、libco)会处理更多细节:异常安全、内存对齐、跨平台兼容、调度器等。但对于理解原理,这个例子足够清晰。

以上就是c++++怎么实现一个简单的协程库_c++手写协程实现原理与示例的详细内容,更多请关注其它相关文章!


# 如何使用  # 怎么联系自适应网站建设  # 甘肃关键词排名上线  # 网站建设推广一年多钱  # 市场营销推广策略会议  # 淘宝客帮推广seo  # 德州提供网站优化费用  # 武汉seo推广排名  # 微信营销推广多少钱  # 无锡营销推广方式  # 集团网站建设内容  # 我们可以  # 是一种  # 网络编程  # 自己的  # 挂起  # linux  # 第三方  # 微软  # 是一个  # str  # win  # macos  # switch  # ios  # c++  # ai  #   # mac  # app  # 操作系统  # windows 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: windows10怎么关闭系统提示音_windows10彻底静音设置方法  机器学习中对数变换预测结果的反向还原  漫蛙2正版漫画站 漫蛙2网页版快速访问入口  拼多多视频播放卡顿如何处理 拼多多视频播放优化技巧  LINUX的perf命令入门_LINUX官方性能分析工具的使用与解读  Lar*el DB::listen 事件中的查询执行时间单位解析  Surface怎么安装系统 微软Surface Pro U盘重装win11教程  Python异步编程实践:使用Binance API构建实时交易数据流  msn官网入口地址手机版 msn官方网站手机最新链接  在J*a中如何在J*a中使用异常机制记录错误日志_异常日志实践经验  优化Django表单:提交验证失败后保留用户输入  如何在CSS中使用visited与link控制链接颜色_visited link伪类配合  优化 Jest 模拟:强制未实现函数抛出错误以提升测试效率  LocoySpider如何部署到云服务器_LocoySpider云部署的远程配置  Mac终端命令大全_Mac常用Terminal指令速查  顺丰快递查询系统 官方正版查询入口  Bing引擎入口最新2025 Bing搜索免费官方登录  12306选座怎么选到临时改签座_12306改签选座策略与步骤  163邮箱注册官网 免费申请163个人邮箱  Django模型中自动计算可用余额的实现方法  C++指针和引用有什么区别_C++内存管理核心概念深度解析  必由学网页版入口 必由学官方平台直接访问  12306几点到几点不能订票? | 官方最新系统维护时间全解析  Bilibili动漫最新防封地址发布-Bilibili动漫2025年最稳正版入口推荐  网易大神账号申诉需要多久_网易大神账号申诉流程说明  KFC套餐升级怎么获取优惠代码_KFC套餐升级活动与优惠代码获取方法  理解Python模块与全局变量的作用域管理  Python模块化编程:有效管理依赖与避免循环引用  Go语言中Map值调用指针接收器方法的限制与应对  海棠账号登录入口_登录海棠账户同步阅读记录  Win11怎么设置鼠标主按键_Win11鼠标左右键功能互换  一加 Nord 5 隐私权限异常_一加 Nord 5 系统安全优化  Django表单提交验证失败后保持字段值不刷新  React Router v6 教程:构建认证保护的私有路由与重定向策略  必由学官网快捷入口 必由学网页版在线学习平台  反效果?《战地6》免费试玩开启后玩家数不升反降  QQ邮箱在线使用入口 QQ邮箱个人账号网页版登录  J*a最大堆Heapify方法修复:索引计算与边界条件深度解析  AO3同人作品网入口 AO3搜索引擎官网永久地址  Win10系统服务哪些可以禁用 Win10安全优化服务列表【干货】  抖音网页版快捷访问 抖音网页版网页版入口操作教程  怎么在html里运行vbs脚本_html中运行vbs脚本方法【教程】  怎样使用“本地安全策略”提升Windows安全性_Secpol.msc配置指南【高手】  Typer应用中灵活处理命令行参数的令牌化与解析  使用J*aScript检测输入元素是否包含在特定类中  Golang如何优化CPU绑定任务分配策略_Golang CPU任务分配优化实践  马斯克:Optimus 人形机器人复数形式为 Optimi  《铁拳8》黑皮辣妹新实机:元气满满的18岁少女!  网站内容防复制粘贴的实现策略与局限性  怎么去除衣服上的口红印_生活小妙招教你用酒精轻松擦除 

搜索