1# Function Flow Runtime串行队列(C++) 2 3<!--Kit: Function Flow Runtime Kit--> 4<!--Subsystem: Resourceschedule--> 5<!--Owner: @chuchihtung; @yanleo--> 6<!--Designer: @geoffrey_guo; @huangyouzhong--> 7<!--Tester: @lotsof; @sunxuhao--> 8<!--Adviser: @foryourself--> 9 10## 概述 11 12FFRT串行队列基于协程调度模型实现,提供高效的消息队列功能,支持异步通信、流量削峰、无锁化状态和资源管理以及架构解耦等多种业务场景。FFRT串行队列支持以下功能: 13 14- **队列创建销毁**,支持创建和销毁队列,创建时可指定队列名称和优先级。每个队列功能上相当于一个单独的线程,队列中的任务相对于用户线程异步执行。 15- **任务延迟**,支持在任务提交时设置延迟时间(`delay`),单位为微秒(`us`)。延迟任务将在`uptime`(提交时刻+延迟时间)后调度执行。 16- **串行调度**,同一队列中的任务按照`uptime`升序排列,严格串行执行。确保队列中上一个任务完成后,下一个任务才会开始执行。 17- **任务取消**,支持根据任务句柄取消未出队的任务。若任务已开始执行或执行完成,则无法取消。 18- **任务等待**,支持根据任务句柄等待任务完成。指定任务完成时,队列中所有`uptime`早于该任务的任务均已执行完成。 19- **任务优先级**,支持在任务提交时设置单个任务的优先级。优先级仅在任务出队后相对于系统其他负载生效,不影响同一队列内任务的串行顺序。若未设置任务优先级,则默认继承队列的优先级。 20 21## 示例:异步日志系统 22 23举例实现一个异步日志系统,主线程将日志任务提交到队列,后台线程从队列中取出任务并写入文件。这种方式既能保证日志的顺序性,又能避免文件写入操作阻塞主线程。 24 25借助FFRT并行化框架API,开发者只需专注于业务逻辑的实现,无需关注异步线程管理、线程安全及调度效率等问题。 26 27用例简化了异常处理和线程安全相关的一些逻辑,实现代码如下所示: 28 29```cpp 30#include <chrono> 31#include <fstream> 32#include <iostream> 33#include <thread> 34#include "ffrt/ffrt.h" // 来自 OpenHarmony 第三方库 "@ppd/ffrt" 35 36class Logger { 37public: 38 Logger(const std::string& filename) 39 { 40 // 创建队列 41 queue_ = std::make_unique<ffrt::queue>("loggerQueue"); 42 43 // 以追加模式打开文件 44 logFile_.open(filename, std::ios::app); 45 if (!logFile_.is_open()) { 46 throw std::runtime_error("Failed to open log file: " + filename); 47 } 48 std::cout << "Log file opened: " << filename << std::endl; 49 } 50 51 ~Logger() { 52 // 销毁队列 53 queue_ = nullptr; 54 55 if (logFile_.is_open()) { 56 logFile_.close(); 57 std::cout << "Log file closed" << std::endl; 58 } 59 } 60 61 // 添加日志任务 62 void log(const std::string& message) { 63 queue_->submit([this, message] { 64 logFile_ << message << std::endl; 65 }); 66 } 67 68private: 69 std::ofstream logFile_; 70 std::unique_ptr<ffrt::queue> queue_; 71}; 72 73int main() 74{ 75 Logger logger("log.txt"); 76 77 // 主线程添加日志任务 78 logger.log("Log message 1"); 79 logger.log("Log message 2"); 80 logger.log("Log message 3"); 81 82 // 模拟主线程继续执行其他任务 83 std::this_thread::sleep_for(std::chrono::seconds(1)); 84 85 return 0; 86} 87``` 88 89## 接口说明 90 91上述样例中涉及到主要的FFRT的接口包括: 92 93| 名称 | 描述 | 94| --------------------------------------------------------------------------------------------------------------------- | -------------- | 95| class [queue](https://gitee.com/openharmony/resourceschedule_ffrt/blob/master/docs/ffrt-api-guideline-cpp.md#queue) | 队列类。 | 96| [sleep_for](https://gitee.com/openharmony/resourceschedule_ffrt/blob/master/docs/ffrt-api-guideline-cpp.md#sleep_for) | 延迟一定时间。 | 97 98> **说明:** 99> 100> - 如何使用FFRT C++ API详见:[FFRT C++接口三方库使用指导](ffrt-development-guideline.md#using-ffrt-c-api-1)。 101> - 使用FFRT C接口或C++接口时,都可以通过FFRT C++接口三方库简化头文件包含,即使用`#include "ffrt/ffrt.h"`头文件包含语句。 102 103## 约束限制 104 105- **避免提交超长任务** FFRT内置进程级队列任务超时检测机制。当串行任务执行时间超过预设阈值(默认30秒)时,系统将打印和上报异常日志,并触发预设的进程超时回调函数(如已配置)。 106- **同步原语使用规范** 在提交给FFRT的任务闭包中,避免使用`std::mutex`、`std::condition_variable`和`std::recursive_mutex`,标准库同步原语会长时间占用FFRT Worker线程。请替换为FFRT提供的同步原语:`ffrt::mutex`、`ffrt::condition_variable`或`ffrt::recursive_mutex`,其用法与标准库相同。 107- **全局变量中的队列管理** 若在全局变量中管理串行队列,随业务进程销毁,测试程序中需注意生命周期解耦问题。在测试用例结束时,需显式释放串行队列,其他资源可随全局变量释放。原因是全局变量在主函数结束后析构,而串行队列的释放依赖于FFRT框架中的其他资源,此时这些资源可能已被销毁。 108