如何设置服务器套接字一次可以接受的连接数限制?我希望能够设置最大连接数,然后一旦达到该限制,客户端连接的任何进一步尝试都将导致超时。到目前为止,我已经尝试过这样的服务器:
sock = socket.socket()
sock.setblocking(0)
sock.bind(address)
sock.listen(0)
connections = []
while True:
readable, writable, exceptional = select.select([sock], [], [])
if readable and len(connections) < MAX_CONNECTIONS:
connection, client_address = s.accept()
connections.append(connection)
# Process connection asynchronously
并为客户提供:
try:
sock = socket.create_connection(self.address, timeout=TIMEOUT)
sock.settimeout(None)
print "Established connection."
except socket.error as err:
print >> sys.stderr, "Socket connection error: " + str(err)
sys.exit(1)
# If connection successful, do stuff
由于程序其余部分的结构,我选择在服务器上使用非阻塞套接字,我不希望改变它。现在,即使达到限制并且服务器停止接受它们,客户端也能够连接到服务器。我该如何解决这个问题?感谢。
答案 0 :(得分:0)
我相信在这里可能会对select()
产生轻微的误解。根据联机帮助页,select()
返回的文件描述符已准备好进行某类IO操作&#34;,其中就绪表示&#34;可以执行相应的IO操作而不会阻塞&#34; ;。
侦听套接字上的相应IO操作是accept()
,如果操作系统已经为您进行了完整的TCP握手,则只能在不阻塞的情况下执行,否则您可能会阻止等待客户端的最终ACK,实例
这意味着只要侦听套接字打开,即使应用程序没有处理,操作系统也会接受连接。 如果要在设定的数字后拒绝连接,则基本上有两个选项:
第二个选项更复杂,需要使用SO_REUSEADDR
选项,这在您的情况下可能不正确。它可能也不适用于所有操作系统,尽管它似乎在Linux上可靠运行。
这里是第二个解决方案的快速草图(因为第一个解决方案很简单)。
def get_listening_socket():
sock = socket.socket()
sock.setblocking(0)
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.bind(('0.0.0.0', 5555))
sock.listen(0)
return sock
sock = get_listening_socket()
LIMIT = 1
conns = {sock}
while True:
readable, writable, exceptional = select.select(conns, [], [])
if sock in readable: # new connection on the listening socket
conn, caddr = sock.accept()
conns.add(conn)
if len(conns) > LIMIT: # ">" because "sock" is also in there
conns.remove(sock)
sock.close()
else: # reading from an established connection
for c in readable:
buf = c.recv(4096)
if not buf:
conns.remove(c)
sock = get_listening_socket()
conns.add(sock)
else:
print("received: %s" % buf)
但是,您可能希望重新考虑为什么您想要首先执行此操作。如果它只是关于在服务器上保存一些内存,那么你可能会过度优化,而应该考虑使用syn-cookies。