当前位置:网站首页>webrtc中的任务队列TaskQueue
webrtc中的任务队列TaskQueue
2022-08-04 05:25:00 【mo4776】
在webrtc中,有一个任务队列TaskQueue
,在视频编码模块中就是通过它是实现编码线程,这篇文章将讲解它的实现和应用场景。
TaskQueue
-任务队列 TaskQueue
的用途是将任务放到到另外一个线程执行,与std::aysnc
功能相似,但是它不能取得执行结果。任务队列是多线程编程下的基础设施,是一种线程间交互手段。
一个TaskQueue
对象就代表了一个任务执行线程,在构造时产生线程,在析构时结束线程。
TaskQeueue
采用Pimpl技术实现,它提供的是接口,内部的impl_
对象封装了具体实现,如下类图,TaskQueue
通过impl_
持有一个TaskQueueBase
类型的对象。
它的构造函数传入一个TaskQueueBase
的实例。
explicit TaskQueue(std::unique_ptr<webrtc::TaskQueueBase,
webrtc::TaskQueueDeleter> task_queue);
TaskQueue
的接口
这里列出它的主要的四个接口: PostTask
,PostDelayedTask
,isCurrent
,Current
PostTask
和PostDelayedTask
一个是将task
放入TaskQueue
立即执行,一个将task
放入TaskQueue
,指定delay
后时间执行。
TaskQueue
类中有两个类型的接口PostTask
和PostDelayedTask
,重载了两种不同类型的参数,一个是传入QueueTask
对象,一个是传入Closure
(Clousre
是模板参数),Closure
使用更加方便,它可以传入一个lambda
表达式。
isCurrent
和Current
isCurrent
可以用于判断任务是否运行在TaskQueue
对象所代表的线程上Current
获取任务线程关联的TaskQueue
对象
这两个方法基于thread_local
类型的变量实现(即线程局部变量),每个线程都有自己的局部变量,在线程内可见,线程间不可见。所以不需要对该类型的变量进行互斥。
首先定义了一个thread_local
类型的TaskQueueBase
指针
thread_local TaskQueueBase* current = nullptr;
在线程执行时,赋值为this
那么isCurrent
的实现如下:
bool IsCurrent() const { return Current() == this; }
Current的实现如下:
TaskQueueBase* TaskQueueBase::Current() {
return current;
}
TaskQueue
对象就代表了一个线程,它可以在任意其它线程使用,在任务中调用IsCurrent()
方法可以判断它是否在TaskQueue
对象所代表的线程上执行。通过这种方式人为的将任务分发到指定的线程,而避免加锁互斥。
TaskQueueBase
具体子类
前面提到 TaskQueue
只是封装接口,TaskQueueBase
才是真正的实现,但它是个抽象接口类,如下图
有4个具体的实现类``TaskQueueStdlib
,TaskQueueWin
,TaskQueueLibevent
,TaskQueueGcd
,分别对应的不同平台的具体实现
- TaskQueueStdlib 跨平台通用版本(linux下的实现,也可以用在windows环境下)
- TaskQueueWin 是windows下的实现
- TaskQueueLibevent 基于libevent实现
- TaskQueueGcd 是IOS下的实现
平台相关的实现主要是在线程的实现(webrtc中并没有使用C++ 11中thread
,是自己封装了thread的API)和定时功能的实现,TaskQueueLibevent
通过libevent
实现定时,TaskQueueWin
通过windows
的消息队列实现定时。TaskQueueStdlib
则是通过条件变量超时机制实现的定时。
TaskQueueBase
对象创建
TaskQueueBase
通过工厂类TaskQueueFactory
来创建具体类型的实例,这个工厂类它还有一个工厂方法,用来创建不同的工厂类实例(有点绕- -!)
task_queue_factory.h
中定义了创建TaskQueue
的工厂类TaskQueueFactory
,有多个工厂方法创建工厂类TaskQueueFactory
。
如下代码,通过工厂类task_queue_factory
创建一个TaskQueueBase
对象,用于构造TaskQueue
对象encoder_queue_
。
//创建一个TaskQueue对象
rtc::TaskQueue encoder_queue_(task_queue_factory->CreateTaskQueue(
"EncoderQueue",
TaskQueueFactory::Priority::NORMAL));
用途
一个TaskQueue
对象产生一个Task执行线程,它本质上是一个异步任务队列,将任务从一个线程分发至任务执行队列,可以起到线程隔离的作用,而减少同步的需求,比如将一系列关联方法放入任务队列可以减少互斥的使用,通过isCurrent
判断方法是否运行在任务执行线程 ,如果是则无需添加同步措施。
可以将TaskQueue
作为特定用途的线程,比如用作编码线程,将编码任务放入线程中。在webrtc中的视频编码就是如此。
代码实例
TaskQueue
的使用比较简单,先通过工厂类创建一个TaskQueueBase
的实例,再根据需求调用PostTask
或PostDelayTask
。在VideoStreamEncoder
类中有个encoder_queue_
的TaskQueue
对象,示例如下:
//创建一个TaskQueue对象
rtc::TaskQueue encoder_queue_(task_queue_factory->CreateTaskQueue(
"EncoderQueue",
TaskQueueFactory::Priority::NORMAL));
//PostTask传入了一个lambada表达式,执行SendKeyFrame方法
encoder_queue_.PostTask([this] {
SendKeyFrame(); });
边栏推荐
猜你喜欢
MySQL log articles, binlog log of MySQL log, detailed explanation of binlog log
乱码解决方案
代码重构:面向单元测试
7.16 Day22---MYSQL (Dao mode encapsulates JDBC)
《看见新力量》第四期免费下载!走进十五位科技创业者的精彩故事
触觉智能分享-SSD20X实现升级显示进度条
一个对象引用的思考
What is the salary of a software testing student?
MySQL日志篇,MySQL日志之binlog日志,binlog日志详解
TensorRT例程解读之语义分割demo
随机推荐
OpenRefine开源数据清洗软件的GREL语言
What is the salary of a software testing student?
FPGA学习笔记——知识点总结
力扣:509. 斐波那契数
渗透测试(PenTest)基础指南
Web Basics and Exercises for C1 Certification - My Study Notes
编程大杂烩(四)
MySQL数据库面试题总结(2022最新版)
8大软件供应链攻击事件概述
MySQL database (basic)
音视频相关基础知识与FFmpeg介绍
7.18 Day23----标记语言
Unity表格配置编辑工具
动态规划总括
7.13 Day20----MYSQL
企业需要知道的5个 IAM 最佳实践
读者让我总结一波 redis 面试题,现在肝出来了
处理List<Map<String, String>>类型
MySQL日志篇,MySQL日志之binlog日志,binlog日志详解
MySQL日期函数