gpt4 book ai didi

python - epoll +非阻塞套接字比阻塞+超时慢吗?

转载 作者:行者123 更新时间:2023-12-03 12:05:07 25 4
gpt4 key购买 nike

我有两个用python编写的简单套接字服务器版本。
第一个版本使用epoll +非阻塞方式,并且似乎比具有阻塞套接字+超时的服务器版本要慢得多。

非阻塞服务器产生10个子代,并且子代在套接字上接受。在这种情况下,所有 child 都将收到EPOLLIN通知,但只有一个 child 可以做
接受后,所有其他子代都将获得EAGAIN,这将被“except-block”忽略。

--- server-nonblocking.py ---

import socket, time, os, select

sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.bind(('127.0.0.1', 10000))
sock.listen(512)
sock.setblocking(False)


for _ in range(0,10):

pid = os.fork()
if pid == 0: #in child
poll = select.epoll()
poll.register(sock.fileno(), select.EPOLLIN)

while True:
events = poll.poll(3) # listening for events with 2 sec timeout
for fileno, event in events:
if event & select.EPOLLIN: # there is data on socket available
print("EPOLLIN in PID: " + str(os.getpid()))
try:
clientsock, addr = sock.accept()
clientsock.close()
print("accepted and closed in PID: " + str(os.getpid()))
except:
pass


# we are in parent process, keep it live
while True:
time.sleep(10)

阻塞服务器还产生10个子代,但在监听套接字上使用超时,而是被except块拦截并忽略超时:

--- server-blocking.py ---
import socket, time, os, select

sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.bind(('127.0.0.1', 10000))
sock.listen(512)
sock.settimeout(5)


for _ in range(0,10):

pid = os.fork()
if pid == 0: #in child
while True:
try:
clientsock, addr = sock.accept()
clientsock.close()
print("accepted and closed in PID: " + str(os.getpid()))
except:
pass


# we are in parent process, keep it live
while True:
time.sleep(10)

这是客户。它仅在环路中连接到服务器并关闭连接。
20秒后。循环将被中断。

--- client.py ---
import socket, time, select, sys

i = 1
td = time.time()
while True:
print("loop " + str(i) + ", time: " + str(time.time() - td))
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.connect(('localhost',10000))
s.setblocking(False)
s.close()
i += 1
if time.time() - td >= 20:
break

这是两个服务器的结果:
blocking:
loop 137670, time: 19.99994468688965

non-blocking:
loop 94051, time: 19.10340452194214

与非阻塞服务器相比,阻塞服务器可以处理更多的连接。当客户端使用非阻塞版本时,
我可以看到循环上的一些延迟。

有人可以解释这种行为吗?为什么在epoll +无阻塞循环中会有一些延迟?

谢谢!!!

最佳答案

epoll()对于以下目的很有用(从手册页epoll(2)):监视多个文件描述符,以查看是否可以在任何文件描述符上进行I/O。

您正在使用epoll()监视一个文件描述符。就上下文切换而言,这增加了很多开销;每个 child 都必须调用epoll_create(),epoll_ctl()和epoll_wait()。接着!每个新的连接都会唤醒他们。接着!他们中的大多数都没有接受。

在阻止版本中,可能只有一个 child 被唤醒。

关于python - epoll +非阻塞套接字比阻塞+超时慢吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26760094/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com