gpt4 book ai didi

如果使用getch,Python套接字recv等待发送?

转载 作者:行者123 更新时间:2023-12-01 01:43:53 24 4
gpt4 key购买 nike

这是一个非常意外的行为:

这是一个教科书经典的短程序,它使用一个线程从套接字流中一个接一个地获取字符并显示它,第二个线程读取输入并通过同一套接字流发送输入。

导入套接字 导入线程 导入getch 导入系统

s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.setsockopt(socket.SOL_TCP, socket.TCP_NODELAY, 1)

def rxWorker():
while(True):
r = s.recv(1)
print(r.decode('ascii'), end='')

def txWorker():
while (True):
i = input('')
s.send(i.encode())

s.connect(('localhost',9999))

threading.Thread(name='Rx', target=rxWorker).start()
threading.Thread(name='Tx', target=txWorker).start()

这适用于在不同终端运行的 netcat 监听器:

nc -l localhost 9999

此时一切正常。线条从一侧发送到另一侧并按预期显示。

现在输入更改为立即输入,因此 python 端在键入字符时发送字符(不等待换行符),如下所示:

导入套接字 导入线程 导入getch 导入系统

s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.setsockopt(socket.SOL_TCP, socket.TCP_NODELAY, 1)

def rxWorker():
while(True):
r = s.recv(1)
print(r.decode('ascii'), end='')

def txWorker():
while (True):
# i = input('')
i = getch.getche()
s.send(i.encode())

s.connect(('localhost',9999))

threading.Thread(name='Rx', target=rxWorker).start()
threading.Thread(name='Tx', target=txWorker).start()

请注意,唯一的变化是读取输入的方式:i = getch.getche()i = input('')

现在行为不同了。

来自 python 端的字符立即正确地出现在 netcat 端。

问题:netcat 端的字符现在不会立即显示在 python 端。实际上,直到一个或几个字符从 python 发送到 netcat 时才会显示。

这很奇怪,有点破坏我的控制流程/:

请指教

系统:Ubuntu 16.04,python 3.5.2

最佳答案

主要问题是您尚未真正启用TCP_NODELAY

这行代码的作用与您想象的不同:

s = socket.socket(socket.AF_INET, socket.TCP_NODELAY )

socket 的参数是:

socket.socket(family=AF_INET, type=SOCK_STREAM, proto=0, fileno=None)

碰巧的是,至少在 *nix 系统上,TCP_NODELAY1,而 SOCK_STREAM 是一个值为 的枚举>1,因此通过传递 TCP_NODELAY 作为您选择 SOCK_STREAM(又名 TCP)的类型。当 socket 库的 future 版本切换到使用枚举作为 sockopt 值时(就像它已经对类型所做的那样),这将成为 TypeError 而不是默默地做一些偶尔发生的意想不到的事情。

如果您想启用 sockopt 值,例如 TCP_NODELAY,则必须调用 setsockopt在 socket 上。像这样:

s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.setsockopt(socket.SOL_TCP, socket.TCP_NODELAY, 1)
<小时/>

顺便说一句,您的代码中至少还有一个明显的错误,尽管它不会导致任何问题:

rxt = threading.Thread(name='Rx', target=rxWorker).start()

Thread.start 返回None。因此,您会看到 rxt(和 txt)变为 None。幸运的是,您实际上并没有对它们做任何事情,但是一旦您这样做,例如rxt.join(),您就会得到一个AttributeError

关于如果使用getch,Python套接字recv等待发送?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51603244/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com