今天在看《Python基础教程(第3版)》的网络编程章节时,有一个通过select实现简单服务器的例程,例程如下:
import socket,select
s=socket.socket()
host=socket.gethostname()
port=1234
s.bind((host,port))
s.listen(5)
inputs=[s]
while True:
rs,ws,es=select.select(inputs,[],[])
for r in rs:
if r is s:
c,addr=s.accept()
print(b'Got connection from',addr)
inputs.append(c)
else:
try:
data=r.recv(1024)
disconnected=not data
except socket.error:
disconnected=True
if disconnected:
print(r.getpeername(),b'disconnected')
inputs.remove(r)
else:
print(data)
执行时,发现在客户端发送数据后,服务端没有接收到数据,同时在data=r.recv(1024)处出现以下错误:
OSError: [WinError 10057] 由于套接字没有连接并且(当使用一个 sendto 调用发送数据报套接字时)没有提供地址,发送或接收数据的请求没有被接受。
经过检查后发现,对于data=r.recv(1024),r为socket类型,但是由于没有执行s.accept(),所以r并没有建立任何连接并接收套接字,所以r.recv(1024)会报错。这里应该是这本书的打印问题,在上面已经定义了c,addr=s.accept(),即建立连接后将套接字标识符保存至c中,并通过c的套接字与客户套接字建立连接。所以对于data=r.recv(1024)只需要改成data=c.recv(1024),同理下面print(r.getpeername(),b'disconnected')应改为print(c.getpeername(),b'disconnected'),inputs.remove(r)改为inputs.remove(c)。