aysncio无法在Windows上读取stdin

blo*_*ley 13 python windows stdin asynchronous python-3.x

我试图在Windows 7 64位和Python 3.4.3上异步读取stdin

我尝试了这个受到SO回答的启发:

import asyncio
import sys


def reader():
    print('Received:', sys.stdin.readline())


loop = asyncio.get_event_loop()
task = loop.add_reader(sys.stdin.fileno(), reader)
loop.run_forever()
loop.close()
Run Code Online (Sandbox Code Playgroud)

然而,它提出了一个OSError: [WInError 100381] An operation was attempted on something that is not a socket.

类似文件的对象是否stdin可以包含在类中以赋予它套接字的API?我已单独提出这个问题,但如果解决方案很简单,请在此处回答.

假设我无法包装类似文件的对象使其成为套接字,我尝试使用流作为这个要点的启发:

import asyncio
import sys


@asyncio.coroutine
def stdio(loop):
    reader = asyncio.StreamReader(loop=loop)
    reader_protocol = asyncio.StreamReaderProtocol(reader)
    yield from loop.connect_read_pipe(lambda: reader_protocol, sys.stdin)


@asyncio.coroutine
def async_input(loop):
    reader = yield from stdio(loop)
    line = yield from reader.readline()
    return line.decode().replace('\r', '').replace('\n', '')


@asyncio.coroutine
def main(loop):
    name = yield from async_input(loop)
    print('Hello ', name)


loop = asyncio.get_event_loop()
loop.run_until_complete(main(loop))
loop.close()
Run Code Online (Sandbox Code Playgroud)

这引起了人们NotImplementedError的注意asyncio.base_events._make_read_pipe_transport

请指教如何读取stdin使用asyncio在Windows ...

Gab*_*ssu 16

NotImplementedError引发异常,因为连接管协程不被支持SelectorEventLoop,这是默认的事件循环设置asyncio.您需要使用a ProactorEventLoop来支持Windows上的管道.然而,这仍然无法工作,因为显然connect_read_pipeconnect_write_pipe功能,不支持stdin/ stdout/ stderr在Windows或文件作为的Python 3.5.1.

stdin使用异步行为读取的一种方法是使用带有循环run_in_executor方法的线程.这是一个简单的参考示例:

import asyncio
import sys

async def aio_readline(loop):
    while True:
        line = await loop.run_in_executor(None, sys.stdin.readline)
        print('Got line:', line, end='')

loop = asyncio.get_event_loop()
loop.run_until_complete(aio_readline(loop))
loop.close()
Run Code Online (Sandbox Code Playgroud)

在该示例中sys.stdin.readline(),该loop.run_in_executor方法在另一个线程内调用该函数.线程保持阻塞状态,直到stdin接收到换行符,同时循环可以自由执行其他协同程序(如果它们存在).

  • @masi`aio_readline()`是非阻塞的,程序本身就是使用`run_until_complete()`来阻塞的.但是你应该只在你的应用程序中使用`await loop.run_in_executor(None,sys.stdin.readline)`部分.其余的只是为了举例说明用法. (2认同)