没有发送就没有接收Python套接字

时间:2017-04-14 22:34:45

标签: python sockets

我从python文档中复制了echo服务器示例,它工作正常。但是当我编辑代码时,它不会将数据发送回客户端,socket.recv()方法在第二次调用时不会返回。

import socket

HOST = ''
PORT = 50007 
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.bind((HOST, PORT))
s.listen(1)

conn, addr = s.accept()
print('Connected by', addr)

while True:
    data = conn.recv(1024)
    if not data: break
conn.sendall(b'ok')
conn.close()

在python文档的原始版本中,while循环略有不同:

while True:
    data = conn.recv(1024)
    if not data: break
    conn.sendall(data)

客户代码:

import socket

HOST = 'localhost'
PORT = 50007
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.connect((HOST, PORT))
s.sendall(b'Hello, world')
data = s.recv(1024)
s.close()
print('Received', repr(data))

2 个答案:

答案 0 :(得分:3)

接收和发送操作的数量必须匹配,因为它们是阻塞的。这是代码的流程图:

Server listen 
Client connect 
Server receive (this waits until a message arrives at the server) [1] 
Client send 'Hello world' (received by [1])
Server receive (because there was data received) [2]
Client receive [3]

因为服务器和客户端现在被阻止,所以没有程序可以继续。

解决方法是删除客户端的接收呼叫,因为您删除了服务器的发送呼叫。

答案 1 :(得分:2)

TCP套接字是数据流。一方的发送呼叫与另一方的接收呼叫之间没有一对一的关联。根据您实施的协议,存在更高级别的关联。在原始代码中,规则是服务器将准确发送它收到的内容,直到客户端关闭连接的传入端。然后服务器关闭了套接字。

随着您的更改,规则发生了变化。现在,服务器不断接收和丢弃数据,直到客户端关闭连接的传入端。然后服务器发送“ok”并关闭套接字。

使用第一个规则的客户端挂起,因为它在关闭套接字之前需要预期的数据。如果它想使用这个新的服务器规则,它必须关闭套接字的传出端以告诉服务器它已完成,然后它就可以获得返回数据。

我已更新客户端和服务器以关闭部分连接,并且客户端还会执行多个recv,以防传入数据碎片化。不太完整的实现似乎适用于小型有效负载,因为您不太可能会出现碎片,但在实际生产代码中可能会崩溃。

服务器

import socket

HOST = ''
PORT = 50007 
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
s.bind((HOST, PORT))
s.listen(1)

conn, addr = s.accept()
print('Connected by', addr)

while True:
    data = conn.recv(1024)
    if not data: break
conn.sendall(b'ok')
conn.shutdown(socket.SHUT_WR)
conn.close()

客户端

import socket

HOST = 'localhost'
PORT = 50007
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.connect((HOST, PORT))
s.sendall(b'Hello, world')
s.shutdown(socket.SHUT_WR)
data = b''
while True:
    buf = s.recv(1024)
    if not buf:
        break
    data += buf
s.close()
print('Received', repr(data))