用户工具


两种IO多路复用方案:Reactor and Proactor

一般情况下,I/O 复用机制需要事件分享器(event demultiplexor [1, 3]). 事件分享器的作用,即将那些读写事件源分发给各读写事件的处理者,就像送快递的在楼下喊: 谁的什么东西送了, 快来拿吧。开发人员在开始的时候需要在分享器那里注册感兴趣的事件,并提供相应的处理者(event handlers),或者是回调函数; 事件分享器在适当的时候会将请求的事件分发给这些handler或者回调函数.

涉及到事件分享器的两种模式称为:Reactor and Proactor [1]. Reactor模式是基于同步I/O的,而Proactor模式是和异步I/O相关的. 在Reactor模式中,事件分离者等待某个事件或者可应用或个操作的状态发生(比如文件描述符可读写,或者是socket可读写),事件分离者就把这个 事件传给事先注册的事件处理函数或者回调函数,由后者来做实际的读写操作。

而在Proactor模式中,事件处理者(或者代由事件分离者发起)直接发起一个异步读写操作(相当于请求),而实际的工作是由操作系统来完成的。发起 时,需要提供的参数包括用于存放读到数据的缓存区,读的数据大小,或者用于存放外发数据的缓存区,以及这个请求完后的回调函数等信息。事件分离者得知了这 个请求,它默默等待这个请求的完成,然后转发完成事件给相应的事件处理者或者回调。举例来说,在Windows上事件处理者投递了一个异步IO操作(称有 overlapped的技术),事件分离者等IOCompletion事件完成[1]. 这种异步模式的典型实现是基于操作系统底层异步API的,所以我们可称之为“系统级别”的或者“真正意义上”的异步,因为具体的读写是由操作系统代劳的。

举另外个例子来更好地理解Reactor与Proactor两种模式的区别。这里我们只关注read操作,因为write操作也是差不多的。

reactor

  1. 某个事件处理者宣称它对某个socket上的读事件很感兴趣;
  2. 事件分离者等着这个事件的发生;
  3. 当事件发生了,事件分离器被唤醒,这负责通知先前那个事件处理者;
  4. 事件处理者收到消息,于是去那个socket上读数据了. 如果需要,它再次宣称对这个socket上的读事件感兴趣,一直重复上面的步骤

proactor

  1. 事件处理者直接投递发一个读操作. 这个时候,事件处理者根本不关心读事件,它只管发这么个请求,它魂牵梦萦的是这个写操作的完成事件。这个处理者很拽,发个命令就不管具体的事情了,只等着别人(一般是操作系统)帮他搞定的时候给他回个话。
  2. 事件分离者等着这个读事件的完成(比较下与Reactor的不同);
  3. 当事件分离者默默等待完成事情到来的同时,操作系统已经在一边开始干活了,它从目标读取数据,放入用户提供的缓存区中,最后通知事件分离者,这个事情我搞完了;
  4. 事件分享者通知之前的事件处理者: 你吩咐的事情搞定了;
  5. 事件处理者这时会发现想要读的数据已经乖乖地放在他提供的缓存区中,想怎么处理都行了。如果有需要,事件处理者还像之前一样发起另外一个写操作,和上面的几个步骤一样。

reactor 分发处理消息,proactor分发处理完成消息

reactor、proactor融合

reactor

  1. 处理者注册感兴趣的消息
  2. 事件分离器通知消息给处理者
  3. 处理者处理消息

proactor

  1. 处理者调用第三方处理消息
  2. 事件分离器通知完成消息给处理者
  3. 处理者得知处理完成后执行下一步操作
  1. 处理者给出感兴趣的事件,并给出如何调用第三方处理事件
  2. 事件分离器等待消息,并调用第三方处理事件,处理完成后通知处理者
  3. 处理者得知自己感兴趣的事情发生并处理完后继续执行下一步操作