gpt4 book ai didi

Python 多处理将子进程的标准输出重定向到 Tkinter 文本

转载 作者:太空宇宙 更新时间:2023-11-04 10:34:52 25 4
gpt4 key购买 nike

我正在尝试使用 Tkinter GUI 来启动子进程并将其 stdout/stderr 输出显示到文本小部件。最初,我认为 sys.stdout 可以通过设置“sys.stdout = text_widget”轻松重定向到文本小部件,但似乎不是。出现错误:“Text instance has no attribute 'flush'”。

网上查了一些解决办法,比如用Queue和子进程通信。但是,由于我的特殊要求,它们都不适合我的情况:

  1. 子进程最好由“multiprocessing.Process”启动,因为使用共享变量会更容易,这使得子进程解决方案可用。
  2. 子进程的代码已经在那里,里面有很多“打印”,所以我不想将它们修改为类似“Queue.put()”之类的东西。

在这种情况下,任何人都可以找到一个解决方案来获取“multiprocessing.Process”的“打印”输出并显示到 Tkinter 文本吗?非常感谢!

我的例子代码如下:

import sys
import time
from multiprocessing import Process
from Tkinter import *

def test_child():
print 'child running'

def test_parent():
print 'parent running'
time.sleep(0.5)
Process(target=test_child).start()

def set_txt(msg):
gui_txt.insert(END, str(msg))
gui_txt.see(END)

if __name__ == '__main__':
gui_root = Tk()
gui_txt = Text(gui_root)
gui_txt.pack()
gui_btn = Button(gui_root, text='Test', command=test_parent)
gui_btn.pack()

gui_txt.write = set_txt
sys.stdout = gui_txt

gui_root.mainloop()

最佳答案

仍然可以使用队列,而不必删除所有的 print 语句。您可以使用依赖于 Processstdout 重定向来执行此操作。下面的解决方案使用 Queue 子类来模拟 stdout。然后,该队列由一个线程监视,该线程查找被泵入文本小部件的新文本。

import sys
import time
from multiprocessing import Process
from multiprocessing.queues import Queue
from threading import Thread
from Tkinter import *

# This function takes the text widget and a queue as inputs.
# It functions by waiting on new data entering the queue, when it
# finds new data it will insert it into the text widget
def text_catcher(text_widget,queue):
while True:
text_widget.insert(END, queue.get())

# This is a Queue that behaves like stdout
class StdoutQueue(Queue):
def __init__(self,*args,**kwargs):
Queue.__init__(self,*args,**kwargs)

def write(self,msg):
self.put(msg)

def flush(self):
sys.__stdout__.flush()


def test_child(q):
# This line only redirects stdout inside the current process
sys.stdout = q
# or sys.stdout = sys.__stdout__ if you want to print the child to the terminal
print 'child running'

def test_parent(q):
# Again this only redirects inside the current (main) process
# commenting this like out will cause only the child to write to the widget
sys.stdout = q
print 'parent running'
time.sleep(0.5)
Process(target=test_child,args=(q,)).start()

if __name__ == '__main__':
gui_root = Tk()
gui_txt = Text(gui_root)
gui_txt.pack()
q = StdoutQueue()
gui_btn = Button(gui_root, text='Test', command=lambda:test_parent(q),)
gui_btn.pack()

# Instantiate and start the text monitor
monitor = Thread(target=text_catcher,args=(gui_txt,q))
monitor.daemon = True
monitor.start()

gui_root.mainloop()

关于Python 多处理将子进程的标准输出重定向到 Tkinter 文本,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23947281/

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