为什么 IO 多路复用要搭配非阻塞 IO
答案:1 悬赏:0 手机版
解决时间 2021-01-08 06:41
- 提问者网友:流星是天使的眼泪
- 2021-01-07 15:41
为什么 IO 多路复用要搭配非阻塞 IO
最佳答案
- 五星知识达人网友:千杯敬自由
- 2021-01-07 17:04
看 Tornado 源码时候,我也产生了同样的疑问。
首先基于 Reactor 模型,socket.fd 已经被放到 ioloop(事件循环) 中,通过多路复用监听到 fd 已可用,并开始调用对应的事件处理函数(handle_read/hanle_accept)。
def handle_read():
socket.setblocking(False)
while True:
try:
data = socket.recv(1024)
except socket.error, e:
if e.args[0] in (errno.EWOULDBLOCK, errno.EAGAIN):
return
raise
def handle_accept():
socket.setblocking(False)
while True:
try:
connection, address = socket.accept()
except socket.error, e:
if e.args[0] in (errno.EWOULDBLOCK, errno.EAGAIN):
return
raise
假如 socket 的读缓冲区中已有足够多的数据,需要调用三次 read 才能读取完。或 ACCEPT 队列已经有三个「握手已完成的连接」。
非阻塞 I/O 的处理方式:循环的 read 或 accept,直到读完所有的数据(抛出 EWOULDBLOCK 异常)。
阻塞 I/O 的处理方式:每次只能调用一次 read 或 accept,因为多路复用只会告诉你 fd 对应的 socket 可读了,但不会告诉你有多少的数据可读,所以在
handle_read/handle_accept 中只能 read/accept 一次,你无法知道下一次 read/accept
会不会发生阻塞。所以只能等 ioloop 的第二次循环,ioloop 告诉你 fd 可用后再继续调用
handle_read/handle_accept 处理,然后再循环第三次。
所以你会发现,后者的处理方式要复杂很多,稍不注意就会阻塞整个进程。
首先基于 Reactor 模型,socket.fd 已经被放到 ioloop(事件循环) 中,通过多路复用监听到 fd 已可用,并开始调用对应的事件处理函数(handle_read/hanle_accept)。
def handle_read():
socket.setblocking(False)
while True:
try:
data = socket.recv(1024)
except socket.error, e:
if e.args[0] in (errno.EWOULDBLOCK, errno.EAGAIN):
return
raise
def handle_accept():
socket.setblocking(False)
while True:
try:
connection, address = socket.accept()
except socket.error, e:
if e.args[0] in (errno.EWOULDBLOCK, errno.EAGAIN):
return
raise
假如 socket 的读缓冲区中已有足够多的数据,需要调用三次 read 才能读取完。或 ACCEPT 队列已经有三个「握手已完成的连接」。
非阻塞 I/O 的处理方式:循环的 read 或 accept,直到读完所有的数据(抛出 EWOULDBLOCK 异常)。
阻塞 I/O 的处理方式:每次只能调用一次 read 或 accept,因为多路复用只会告诉你 fd 对应的 socket 可读了,但不会告诉你有多少的数据可读,所以在
handle_read/handle_accept 中只能 read/accept 一次,你无法知道下一次 read/accept
会不会发生阻塞。所以只能等 ioloop 的第二次循环,ioloop 告诉你 fd 可用后再继续调用
handle_read/handle_accept 处理,然后再循环第三次。
所以你会发现,后者的处理方式要复杂很多,稍不注意就会阻塞整个进程。
我要举报
如以上回答内容为低俗、色情、不良、暴力、侵权、涉及违法等信息,可以点下面链接进行举报!
点此我要举报以上问答信息
大家都在看
推荐资讯