标签导航:

python asyncio tcp服务器:telnet连接即断开的原因是什么及如何解决?

使用Python asyncio构建TCP服务器时,Telnet客户端连接后立即断开的原因及解决方法

本文分析一个基于Python asyncio库的简单TCP服务器,在使用Telnet客户端连接时出现“connection closed by foreign host”错误,并提供解决方案。

问题描述:

代码示例中,作者使用asyncio库编写了一个简单的TCP服务器,监听本地8888端口。然而,使用Telnet客户端连接该服务器后,会立即断开连接,并显示“connection closed by foreign host”。

原始代码:

import asyncio
from asyncio.base_events import server


async def handle_client(reader, writer):
    while True:
        data = await reader.readline()
        if not data:
            break
        writer.write(data)
    writer.close()


async def main():
    server: server = await asyncio.get_running_loop().create_server(
        handle_client, '127.0.0.1', 8888)

    async with server:
        await server.serve_forever()

asyncio.run(main())

问题根源在于handle_client函数中,writer.write(data)之后缺少await writer.drain()。writer.write仅将数据写入缓冲区,并未保证数据实际发送到客户端。writer.drain()用于等待缓冲区数据完全发送,这是确保客户端正确接收数据的关键。服务器在未等待数据发送完成就结束处理,导致客户端未收到数据而断开连接。

解决方案:

为了解决这个问题,需要在writer.write后添加await writer.drain(),确保每次写入后数据都完全发送。此外,添加欢迎信息可以提升用户体验。改进后的handle_client函数如下:

async def handle_client(reader, writer):
    welcome_message = "Welcome to the server!
"
    writer.write(welcome_message.encode())
    await writer.drain()

    while True:
        data = await reader.read(100)
        if not data:
            break
        writer.write(data)
        await writer.drain()
    writer.close()

通过添加await writer.drain(),保证了数据完全发送,解决了Telnet客户端连接即断开的问题。reader.read(100)比reader.readline()更适合处理Telnet输入,因为Telnet输入可能不以换行符结尾。