【问题标题】:Python socket recv waits for send if using getch?如果使用getch,Python套接字recv等待发送?
【发布时间】:2018-07-30 22:43:38
【问题描述】:

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

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

导入套接字 导入线程 导入 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() vs i = input('')

现在行为不同了。

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

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

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

请指教

系统:Ubuntu 16.04,python 3.5.2

【问题讨论】:

  • 根据 abarnert 的注释对示例代码进行了一些更改。问题,如描述的那样持续存在。

标签: python python-sockets


【解决方案1】:

主要问题是您实际上没有启用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 库的未来版本切换到对sockopt 值使用枚举时(就像它已经对类型所做的那样),这将成为TypeError,而不是默默地做一些意外的事情,偶尔会发生在某种工作中。

如果要启用sockopt 值,如TCP_NODELAY,则必须在套接字上调用setsockopt。像这样:

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

【讨论】:

  • 感谢 cmets。我根据cmets更改了问题,以使问题更清楚。不幸的是,问题与标志无关,并且仍然存在。
猜你喜欢
  • 2011-04-04
  • 2013-11-16
  • 2020-09-05
  • 2018-09-16
  • 1970-01-01
  • 2016-09-29
  • 2013-12-15
  • 2017-12-10
  • 2014-11-01
相关资源
最近更新 更多