根据这些评论,我对最有可能的问题有一个猜测,但至少有50%的可能性我猜错了,在这种情况下……告诉我,我会删除答案。在

我猜你是在尝试使用流套接字,就好像它是一个消息序列一样。这是网络编程新手中非常常见的问题。在

想象一下发送者会这样做:data = pickle.dumps(object);

self.sock.sendall(data)

接受者会这样做:

^{pr2}$

在简单的测试中,这可能在99%的时间内有效,但在实际应用中却行不通。您将收到部分消息,或在一次呼叫中收到多条消息,或上述内容的一些有趣组合(如消息2的一半、消息3的全部和消息4的三分之一)。在

因此,您将向loads传递一条部分消息,并返回一个错误,告诉您它不是一个完整的pickle。在

这并不是因为任何东西都坏了,这才是所谓的工作原理。(TCP)套接字是一个流:一个字节序列,而不是一个消息序列。任何你想在上面建立的结构,你都必须在数据中构建。在

这意味着您必须设计和实现一个协议,以某种方式知道每个消息何时结束。最简单的协议可能是行(显然只有在消息不能有未经转义的换行符时才有用)和netstrings,但是任何能让您清楚地看到某些数据并说“这是消息0,这是消息1,等等”的任何东西都可以工作。在

通常,这意味着将接收到的数据附加到某个缓冲区中,并循环该缓冲区中的消息。例如,使用线条,而不是:

while True:
line = sock.recv(4096)
do_stuff(line)
…你需要这个:rdbuf = ''
while True:
rdbuf += sock.recv(4096)
lines = rdbuf.split('\n')
rdbuf = lines[-1]
for line in lines[:-1]:
dostuff(line)

如果你仔细想想,这和文件没什么不同。想象一下这个代码:with open('foo.data', 'wb') as f:

f.write('123')
f.write('45')
with open('foo.data', 'rb') as f:
while True:
number = f.read()

这将读到'12345',而不是{}。如果你想得到'123',你需要一些方法知道只读取3个字节。插入一个长度前缀,或添加空格作为分隔符,或只是外部知道第一个数字总是3位数长…任何都可以,但您必须做一些的事情。