14

我一直在编写一些代码来替换一些现有的:

while(runEventLoop){
  if(select(openSockets, readFDS, writeFDS, errFDS, timeout) > 0){
    // check file descriptors for activity and dispatch events based on same 
  }
} 

套接字读取代码。我想将其更改为使用 GCD 队列,以便我可以使用 dispatch_async 将事件弹出到队列中,而不是维护“必须在下一次迭代时调用”数组。我也已经在使用 GCD 队列来 /contain/ 这个特定的操作,因此希望将其转移到更自然的 GCD 调度形式。(不是独占串行队列的 while() 循环)

但是,当我尝试将其重构为依赖于从绑定到套接字描述符上的 DISPATCH_SOURCE_TYPE_READ 和 DISPATCH_SOURCE_TYPE_WRITE 的事件处理程序中触发的调度源的表单时,依赖于此调度的库代码停止工作。我的第一个假设是我误解了 DISPATCH_SOURCE_TYPE_READ 和 DISPATCH_SOURCE_TYPE_WRITE 的使用——我曾假设它们会产生与使用这些套接字描述符调用 select() 大致相同的行为。

我是否误解了 GCD 调度来源?或者,关于重构,我是否在不适合的情况下使用它?

4

1 回答 1

3

对您的问题的简短回答是:没有。没有区别,GCD 调度源select()都做同样的事情:它们通知用户发生了特定的内核事件或特定条件成立。

请注意,在 Mac 或 iOS 设备上,您不应使用select(),而应使用更高级的kqueue()kevent()(或kevent64())。

您当然可以将代码转换为使用 GCD 调度源,但您需要注意不要破坏依赖于此的其他代码。因此,这需要对整个代码处理信号、文件描述符、套接字和所有其他低级内核事件进行全面检查。

可能是一个更简单的解决方案可能是维护原始代码,只需在对事件做出反应的部分添加 GCD 代码。在这里,您根据特定的事件类型在不同的队列上分派事件。

于 2011-03-04T15:14:31.483 回答