【问题标题】:gevent.StreamServer and non-blocking raw_input()?gevent.StreamServer 和非阻塞 raw_input()?
【发布时间】:2012-05-15 13:09:14
【问题描述】:

出于测试目的,我使用gevent.StreamServer 编写了一个简单的tcp 服务器。为了让我向某些客户端发送响应,我需要一种非阻塞方式来处理通过raw_input() 的输入,最好不使用线程。

经过一番谷歌搜索后,我偶然发现了这个问题:How to make non-blocking raw_input when using eventlet.monkey_patch() and why it block everything, even when executed on another thread?

我已经编写了以下内容,它完全符合我的要求,但我认为有更好的方法。有人能指出我正确的方向吗?此外,感谢 try/except 没有捕获 KeyboardInterrupt 的想法。

import select
from gevent.monkey import patch_all
from gevent.server import StreamServer

patch_all(os=True, select=True)

def raw_input(message):
    """ Non-blocking input from stdin. """
    sys.stdout.write(message)

    select.select([sys.stdin], [], [])
    return sys.stdin.readline()

def main():
    """ Run the server, listen for commands """

    server = StreamServer(("0.0.0.0", 6000), handle)
    print "Starting server"
    gevent.signal(signal.SIGTERM, server.close)
    gevent.signal(signal.SIGQUIT, server.close)
    gevent.signal(signal.SIGINT, server.close)

    server.start()
    while True:
        try:
            a = raw_input("")
            if a:
                print "Received %s" % a
            gevent.sleep(0)
        except KeyboardInterrupt:
            print "Received a shutdown signal, going down ..."
            server.stop()
            sys.exit(0)

if __name__ == "__main__":
    main()

编辑:我已经重写了部分代码,现在我明白了 main() 函数的愚蠢之处。我会把它作为编辑发布,以防有​​人偶然发现这个问题并对如何做有更好的想法。

from gevent.signal import signal

def get_console_input():
    """ Non-blocking console input to the server """

    select.select([sys.stdin], [], [])
    # There's quite a bit of code here but it is input handling so 
    # for shortness's sake   I've snipped everything and return the line read.
    return sys.stdin.readline()

def exit(server):
    """ Quit the server gracefully """

    print "Received shutdown signal, going down. """
    server.close()
    sys.exit(0)

def main():
    """ The main function. Create and run the server, listen for commands and
   append any command to the server so it can send them on to the clients """

    # Myserver is a class which inherits from gevent.server.StreamServer.
    # Myserver could just as well be replaced by gevent.server.StreamServer.

    server = MyServer(("0.0.0.0", PORT))
    print "Starting server"

    # Add some signal handlers to quit the server gracefully.
    gevent.signal(signal.SIGTERM, exit, server)
    gevent.signal(signal.SIGQUIT, exit, server)
    gevent.signal(signal.SIGINT, exit, server)

    # Server started in async mode
    server.start()

    while True:
        get_console_input()
        gevent.sleep(0)

if __name__ == "__main__":
    main()

【问题讨论】:

    标签: python gevent


    【解决方案1】:

    我发现的最简单的方法是使用gevent.socket.wait_read 等到sys.stdin 可以读取:

    wait_read(sys.stdin.fileno())
    return sys.stdin.readline()
    

    我还围绕文件描述符编写了一个小包装器,以赋予它们非阻塞 .read().write() 方法:https://gist.github.com/2915875

    【讨论】:

    • 谢谢,似乎正是我要找的东西!
    • 在 OSX 中运行良好,但在 Windows 中似乎无法正常运行。 wait_read 引发“IOError: [Errno 0] No error”
    • 目前gevent没有wait_read和wait_write方法
    猜你喜欢
    • 2016-07-06
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2014-10-16
    • 2014-03-23
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多