首页 > 代码库 > 基于纤程(Fiber)实现C++异步编程(一):原理及示例

基于纤程(Fiber)实现C++异步编程(一):原理及示例

纤程(Fiber)和协程(coroutine)是差不多的概念,也叫做用户级线程或者轻线程之类的。Windows系统提供了一组API用户创建和使用纤程,本文中的库就是基于这组API实现的,所以无法跨平台使用,非Windows程序员可以闪人了,当然如果有兴趣可以继续看下去,找个第三方的协程库封装一下,也能实现相同的效果。关于纤程更详细的信息可以查阅MSDN

纤程的概念中有两个关键点:

  1. 纤程拥有独立的栈空间和寄存器环境
  2. 纤程在用户态实现调调度,也就是说完全由程序员控制;

下面的图表示了几个纤程相互切换的过程,注意每个纤程都有独立的栈,并且通过SwitchToFiber函数切换到其他纤程:

技术分享

 

作为对比,我们可以看一下函数调用过程中的堆栈变化情况,下面是示意图,表示了func1 -> func2 -> func3 这种常见的函数嵌套调用关系:

技术分享 

 

每一次函数调用都会创建一个新的栈帧(stack frame),合起来就构成整个调用栈,函数返回时其栈帧也随之释放。对于函数调用,我们可以确定的一点是(在不抛出异常的情况下)被调用函数执行完毕后一定会在调用点返回并继续执行下一条语句。但纤程之间的调用(切换)却不同,一个纤程可以在任意位置切换到其他纤程,并且可能永远都不会再切换回来,也可能从其他任意纤程(不必是刚刚切换到的)切换回来,前面的示意图描述的只是一种非常简单的情况,实际的情况可能非常复杂,复杂到导出都是跳来跳去的箭头理也理不清。在纤程间切换,有点像用加强版的goto,用的时候固然很爽,但后续的维护却是个麻烦。

所以就像用while/for/switch-case代替goto一样,我们也需要封装一组新的API来代替对操作系统API的直接调用。一方面,在封装过程中我们可以对纤程的行为(实际是程序员的行为)施加一些安全约束,使得更容易写出安全的代码或者更不容易写出不安全的代码;另一方面,从goto到while/switch等过程控制语句实际上是一种抽象层次的提升,对大部分常见需求后者用起来更方便,写出的代码也更简洁易懂,类似的从系统API到新的封装API或者封装类也是抽象层次的提高,可以更加方便的应用在各种业务场景。最后,直接使用系统API需要写很多维护纤程的辅助代码,这类代码通常重复而又分散到业务代码的各个角落,进一步降低了程序的可读性和提高了维护难度,封装也是为了解决这个问题。

好了,废话说完了,我们先上一段代码尝尝鲜:

 1     const int RUN_TIMES = 5; 2  3     int number = 0; 4     bool shutdown = false; 5  6     Fiber fib([&number, &shutdown] 7     { 8         while (!shutdown) 9         {10             number++;11             Fiber::yield();             // A:控制权移交到主纤程12         }13     });14 15     for (int i = 0; i < RUN_TIMES; i++)16     {17         fib.resume();                    // B: 切换到子纤程执行18     }19     20     printf("number = %d\r\n", number);        

这里先创建了一个纤程实现number变量累加的功能,然后在for循环中执行(姑且用这个词)最终得到正确的结果。AB两处代码分别实现了纤程的切换,实际上是封装了对SwitchToFiber的调用,注意两个函数调用细节上的不同:resume是普通成员函数表示切换到对象包装的纤程中执行,yield是静态成员函数表示控制权移交给调用者纤程,大家可以思考下为什么有这样的差别。

下面是用纤程实现生产者-消费者模型的代码:

 1     int product_count = 0; 2     bool is_end_time = false; 3  4     const int RUN_TIMES = 3; 5  6     // 生产者纤程 7     Fiber fib_producer([&is_end_time, &product_count] 8     { 9         srand((unsigned)time(NULL));10 11         while (!is_end_time)12         {13             int new_product_count = (int)((double)rand() / RAND_MAX * 10) + 1;14             product_count += new_product_count;15 16             printf("[producer] create new products: %d\r\n", new_product_count);17 18             Fiber::yield();19         }20 21         printf("[producer] off duty.\r\n");22     });23 24     // 消费者纤程的执行函数25     auto consumer_proc = [&is_end_time, &product_count](const int seq_number)26     {27         int total_count = 0;28 29         while (!is_end_time)30         {31             if (product_count > 0)32             {33                 product_count--;34                 total_count++;35                 printf("[consumer %d] got 1 product, total got %d, remain %d\r\n", seq_number, total_count, product_count);36             }37 38             Fiber::yield();39         }40 41         printf("[consumer %d] off duty.\r\n", seq_number);42     };43 44     const int CONSUMER_COUNT = 3;45     int consumer_seq_number = 0;46 47     // 创建消费者纤程数组48     std::vector<Fiber> consumer_array(CONSUMER_COUNT);49     std::for_each(consumer_array.begin(), consumer_array.end(), [&](Fiber& item){ item = Fiber([&]{ consumer_proc(consumer_seq_number); }); consumer_seq_number++; });50 51     consumer_seq_number = 0;52 53     for (int i = 0; i < RUN_TIMES; i++)54     {55         fib_producer.resume();56 57         while (product_count > 0)58         {59             consumer_array[consumer_seq_number].resume();60             consumer_seq_number = (consumer_seq_number + 1) % CONSUMER_COUNT;61         }62     }63 64     is_end_time = true;65 66     // 等待纤程结束67     Fiber::await_all(consumer_array);68     Fiber::await(fib_producer);

程序末尾出现了await和await_all两个新的方法可以先不用管,不影响主要逻辑。由于所有纤程都是在同一个线程中运行的所以无需加锁,这也是使用纤程的一个重要好处。

 

限于篇幅,这次就只写这么多了,更多的内容将放到后面的帖子中,总计还要写四、五篇的样子。但代码实际上已经写完了,急性子的同学可以直接到这个地址看代码:

https://code.csdn.net/xrunning/fiber

 

基于纤程(Fiber)实现C++异步编程(一):原理及示例