您可以使用手柄選擇I/O複用:
https://docs.python.org/3/library/selectors.html
嘗試了這一點:
#! /usr/bin/python3
import sys
import fcntl
import os
import selectors
# set sys.stdin non-blocking
orig_fl = fcntl.fcntl(sys.stdin, fcntl.F_GETFL)
fcntl.fcntl(sys.stdin, fcntl.F_SETFL, orig_fl | os.O_NONBLOCK)
# function to be called when enter is pressed
def got_keyboard_data(stdin):
print('Keyboard input: {}'.format(stdin.read()))
# register event
m_selector = selectors.DefaultSelector()
m_selector.register(sys.stdin, selectors.EVENT_READ, got_keyboard_data)
while True:
sys.stdout.write('Type something and hit enter: ')
sys.stdout.flush()
for k, mask in m_selector.select():
callback = k.data
callback(k.fileobj)
上面的代碼將保持在線路
for k, mask in m_selector.select():
直到註冊事件發生時,返回一個selector_key實例(k)和監視的事件的掩模。
在上面的例子中,我們只登記了一個事件(輸入按鍵):
m_selector.register(sys.stdin, selectors.EVENT_READ, got_keyboard_data)
鍵實例定義爲選擇器如下:
abstractmethod register(fileobj, events, data=None)
因此,寄存器方法集ķ。數據作爲我們的回調函數got_keyboard_data,並調用它時,輸入按鍵:
callback = k.data
callback(k.fileobj)
一個更完整的示例(希望更多有用)將複用來自用戶的標準輸入數據從進來的連接網絡:
import selectors
import socket
import sys
import os
import fcntl
m_selector = selectors.DefaultSelector()
# set sys.stdin non-blocking
def set_input_nonblocking():
orig_fl = fcntl.fcntl(sys.stdin, fcntl.F_GETFL)
fcntl.fcntl(sys.stdin, fcntl.F_SETFL, orig_fl | os.O_NONBLOCK)
def create_socket(port, max_conn):
server_addr = ('localhost', port)
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
server.setblocking(False)
server.bind(server_addr)
server.listen(max_conn)
return server
def read(conn, mask):
global GO_ON
client_address = conn.getpeername()
data = conn.recv(1024)
print('Got {} from {}'.format(data, client_address))
if not data:
GO_ON = False
def accept(sock, mask):
new_conn, addr = sock.accept()
new_conn.setblocking(False)
print('Accepting connection from {}'.format(addr))
m_selector.register(new_conn, selectors.EVENT_READ, read)
def quit():
global GO_ON
print('Exiting...')
GO_ON = False
def from_keyboard(arg1, arg2):
line = arg1.read()
if line == 'quit\n':
quit()
else:
print('User input: {}'.format(line))
GO_ON = True
set_input_nonblocking()
# listen to port 10000, at most 10 connections
server = create_socket(10000, 10)
m_selector.register(server, selectors.EVENT_READ, accept)
m_selector.register(sys.stdin, selectors.EVENT_READ, from_keyboard)
while GO_ON:
sys.stdout.write('>>> ')
sys.stdout.flush()
for k, mask in m_selector.select():
callback = k.data
callback(k.fileobj, mask)
# unregister events
m_selector.unregister(sys.stdin)
# close connection
server.shutdown()
server.close()
# close select
m_selector.close()
您可以使用兩個終端進行測試。 第一終端:
$ python3 test.py
>>> bla
打開另一個終端並運行:
$ nc localhost 10000
hey!
回到第一
>>> qwerqwer
結果(觀察到的主終端上):
$ python3 test.py
>>> bla
User input: bla
>>> Accepting connection from ('127.0.0.1', 39598)
>>> Got b'hey!\n' from ('127.0.0.1', 39598)
>>> qwerqwer
User input: qwerqwer
>>>
[也許](http://stackoverflow.com/questions/ 8416586 /關閉緩衝)一個[重複](http://stackoverflow.com/questions/107705/python-output-buffering) –
在stdin上的非阻塞要麼不起作用,要麼不可靠地工作。你允許使用線程/多處理嗎?應該起作用的原因 –