我在windows10上使用python3.6和asyncio来制作一个简单的服务器。我有问题让服务器以我想要的方式工作。 但是使用套接字接口实现的同一服务器可以正常工作。 这是工作服务器
import socket
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
s.bind(('', 3201))
s.listen(1)
conn, addr = s.accept()
with conn:
print('Client connection: {}:{}'.format(*addr))
while True:
data = conn.recv(1024)
if data:
print('recv {} bytes'.format(len(data)))
else:
print('no data')
break
print('connection closed')
我用nc命令测试它。 e.g。
nc 10.68.100.32 3201
abc
ab
a
^C
服务器打印:
Client connection: 10.68.101.87:55243
recv 4 bytes
recv 3 bytes
recv 2 bytes
no data
connection closed
但是我不能用asyncio做同样的服务器。 这是一个协议版本:
import asyncio
class ListenServer(asyncio.Protocol):
def connection_made(self, transport):
address = transport.get_extra_info('peername')
print('connection made by client:{}:{}'.format(*address))
def data_deceived(self, data):
print('received {}'.format(len(data)), flush=True)
def eof_received(self):
print('print received EOF')
def connection_lost(self, error):
if error:
print('ERROR: {}'.format(error))
else:
print('connection closed')
event_loop = asyncio.get_event_loop()
factory = event_loop.create_server(ListenServer, '', 3201)
server = event_loop.run_until_complete(factory)
event_loop.run_forever()
print('closing server')
server.close()
对于与上述服务器写入相同的nc测试序列:
connection made by client:10.68.101.87:55247
print received EOF
connection closed
我也失败了服务器的asyncio coroutine API版本:
import asyncio
async def just_read(reader, writer):
cli_host, cli_port = writer.get_extra_info('peername')
print('Client connected {}:{}'.format(cli_host, cli_port))
while True:
data = await reader.read()
if data:
print('received {} bytes'.format(len(data)), flush=True)
else:
print('closing')
writer.close()
return
event_loop = asyncio.get_event_loop()
coro = asyncio.start_server(just_read, '', 3201, loop=event_loop)
server = event_loop.run_until_complete(coro)
event_loop.run_forever()
server.close()
print('Connection closed')
在这种情况下的输出是一个小小的差异:
Client connected 10.68.101.87:55253
received 9 bytes
closing
在这种情况下,当我用^ C关闭nc时,我得到所有数据, 并且服务器继续,以便我可以再次连接。
我希望服务器读取并打印出客户端写的内容, 只要客户端连接,它应连接一次并保持连接。 如何使用asyncio实现这一点,我应该使用协议或协程版本,这有关系吗?
这只是我希望服务器执行的几项任务中的第一项,也可能是最简单的。 此处未示出的其他任务是开始定期发送虚拟消息 并发送不同消息的响应。
答案 0 :(得分:1)
协议代码包含拼写错误,它将方法命名为data_deceived
而不是data_received
。方法名称更正后,输出将与阻止示例中的输出相同。
在协程版本中,问题是reader.read()
指示asyncio收集数据直到文件结束,并在准备好后提供。将其更改为reader.read(1024)
,与阻止版本中的sock.recv(1024)
类似,修复代码。