协程退出与智能指针

协程退出与智能指针

该问题是由于智能指针的引用计数问题引起的。fiber使用的智能指针无法析构导致fiber类无法析构。

问题描述

实现sylar协程模块时,遇到了协程退出的问题。 sylar的协程是有栈协程,且只支持主协程与子协程之间的切换。 当一个子协程任务函数执行完毕,需要退出时,需要yeildToHold(swapOut)将控制权转回主协程。但如果是子协程主动yeild,将导致任务函数无法完全完成,对应的协程类的智能指针计数永远大于等于1。如代码所示:

cpp 复制代码
// 协程 执行函数
void Fiber::MainFunc() {
    Fiber::ptr cur_fiber = GetThisFiber();
    M_SYLAR_ASSERT(cur_fiber);

    // 任务函数执行逻辑
    // ...

    // 协程退出逻辑
    cur_fiber->swapOut();

    // 添加assert防止销毁协程重新进入
    M_SYLAR_ASSERT2(false, "fiber has already been destoried, but still swaped in.");
}

如代码所示,Fiber::MainFunc()中持有的cur_fiber是当前协程的智能指针。由于cur_fiber已经swapOut切出了,导致智能指针(cur_fiber)计数永远加一,类永远无法析构。

问题分析

该问题是由于智能指针的引用计数问题引起的。使用的智能指针无法析构导致fiber类无法析构。需要解决函数内的智能指针析构的问题。

问题解决

既然是智能指针引发的问题,那么不用智能指针不就解决了。如代码所示:

cpp 复制代码
// 协程 执行函数
void Fiber::MainFunc() {
    Fiber::ptr cur_fiber = GetThisFiber();
    M_SYLAR_ASSERT(cur_fiber);

    // 任务函数执行逻辑
    // ...

    // 协程退出逻辑
    auto raw_ptr = cur_fiber.get();     // 获取类的指针
    cur_fiber.reset();                  // 使fiber对象引用计数减一
    raw_ptr->swapOut();

    M_SYLAR_ASSERT2(false, "fiber has already been destoried, but still swaped in.");
}

通过使用raw_ptr裸指针,同时重置cur_fiber使得引用计数减一,即可解决智能指针无法销毁的问题。

新的问题

但是这样不会导致raw_ptr变成野指针吗? 其实不会的。

cpp 复制代码
void Fiber::MainFunc() {
    Fiber::ptr cur_fiber = GetThisFiber();
    M_SYLAR_ASSERT(cur_fiber);

    // 任务函数执行逻辑
    // ...

    // runner_fiber 执行完毕
    // 协程退出逻辑
    auto raw_ptr = cur_fiber.get();     // 获取类的指针
    cur_fiber.reset();                  // 使fiber对象引用计数减一
    raw_ptr->swapOut();

    M_SYLAR_ASSERT2(false, "fiber has already been destoried, but still swaped in.");
}

void runner_fiber () {
    m_sylar::Logger::ptr self_logger = M_SYLAR_GET_LOGGER_ROOT();
    m_sylar::Logger::ptr system_logger = M_SYLAR_LOG_NAME("system");
    M_SYLAR_LOG_INFO(system_logger) << "runner_fiber begin";
    m_sylar::Fiber::YieldToHold();          // 交出执行权
    M_SYLAR_LOG_INFO(system_logger) << "runner_fiber end";
}

void runner_thread() {
    // ...
    m_sylar::Fiber::ptr newFiber (new m_sylar::Fiber(runner_fiber));        // 这里有子fiber的一个智能指针。
    newFiber->swapIn();         // 第一次进入fiber
    M_SYLAR_LOG_INFO(self_logger) << "runner_thread swapIn";
    newFiber->swapIn();         // 第二次进入fiber,此次进入后newFiber应该执行完毕

    // ...
}

runner_fiber 执行完毕时,runner_thread中仍持有一个newFiber,所以cur_fiber.reset(); 并不会导致子协程类的析构。 使用gdb调试也印证了这一点。 如果从一开始就用裸指针也同样可以解决问题。

结论与反思

从此次遇到的问题可以做个总结:对于获取的智能指针使用上述方法,不会导致野指针的产生(除非你把它传参出去了)。

相关推荐
心勤则明10 分钟前
Spring AI Alibaba Skills 的渐进式披露与热更新实战
java·后端·spring
西柚小萌新14 分钟前
【人工智能:Agent】--OpenClaw设计架构解析
运维·服务器·架构
小程故事多_8015 分钟前
AI Coding 工程化革命,Superpowers 管流程,ui-ux-pro-max 管质感
人工智能·ui·架构·aigc·ai编程·ux·claude code
金融数据出海43 分钟前
java对接美股股票api涵盖实时行情、K 线、指数等核心接口。
后端
好运的阿财43 分钟前
“锟斤拷”问题——程序中用powershell执行命令出现中文乱码的解决办法
linux·前端·人工智能·机器学习·架构·编辑器·vim
认真的小羽❅1 小时前
从入门到精通:Spring Boot 整合 MyBatis 全攻略
spring boot·后端·mybatis
提子拌饭1331 小时前
开源鸿蒙跨平台Flutter开发:AR太空探索应用
flutter·华为·架构·开源·harmonyos·鸿蒙
小陈工1 小时前
Python Web开发入门(十八):跨域问题解决方案——从“为什么我的请求被拦了“到“我让浏览器乖乖听话“
开发语言·python·机器学习·架构·数据挖掘·回归·状态模式
墨雪遗痕1 小时前
工程架构认知(二):从 CDN 到 Keep-Alive,理解流量如何被“消化”在系统之外
java·spring·架构
摆烂工程师1 小时前
教你如何查询 Codex 最新额度是多少,以及 ChatGPT Pro、Plus、Business 最新额度变化
前端·后端·ai编程