多路事件循环(Multiple event loops)¶
你也可以在同一个线程中使用多路事件循环, 但意义并不大, 因为调用其中某一个事件循环的
uv_run
会使得程序阻塞, 其他事件循环不能同时运行.
不过也可以小心地通过多次调用 uv_run_once() 来完成一些有趣的工作.
多路事件循环的形式(Modality)¶
你可以在你的程序中按照相对标准方式使用多路事件循环, 即第二个事件循环先暂停第一个事件循环, 直到发生了动作(用户按下了 Return(Enter) 键, 或者发生了新的事件等).
各线程拥有自己的事件循环(One loop per thread)¶
One loop per thread
才是多路事件循环的’标准模型’, 它和我们在
进程 一章中介绍的产生多个进程方式并无太大区别.
使用两个事件循环来同步(sing two loops for synchronization)¶
也有一些比较特殊的应用场景需要两个事件循环来进行同步(而不是使用条件变量), 在 node-taglib 库中我就使用了这种同步机制. 具体的应用场景如下:
- 主线程 main thread 通过 uv_queue_work 在工作线程中调用一个阻塞函数.
- 工作线程 worker thread 调用一个自定义函数, 并且自定义函数必须在主线程中运行.
- 工作线程一直等待, 直到函数返回.
如果使用条件变量来实现:
- 工作线程并不直接调用自定义函数, 而是创建一个
uv_async_t
句柄, 然后由该句柄的回调函数调用自定义函数. - 初始化条件变量.
- 使用 uv_async_send() 获得主线程(运行事件循环的线程)并由主线程调用该函数.
- 等待条件变量的通知.
- 回调函数调用自定义函数, 并向条件变量发送通知, 使得工作线程得以继续下去.
而事件循环的实现方式为:
- 在工作线程中创建一个新的事件循环.
- 将 uv_async_t 与事件循环相关联.
- 将该句柄通过 uv_async_t 结构 data 字段传递给主线程.
- uv_run() 运行事件循环, 此时程序会阻塞, 因为异步句柄自增了它的 refcount.
- 主线程的回调函数再调用自定义函数,然后调用 uv_async_send 向事件循环的异步句柄发送消息.
- 异步句柄的回调函数关闭自己, 然后事件循环的引用计数减为 0, uv_run 返回, 工作线程得以继续下去.