gpt4 book ai didi

python - 将 sys.stdout 重定向到 Tkinter.Text 小部件时出现段错误

转载 作者:太空狗 更新时间:2023-10-30 00:57:59 24 4
gpt4 key购买 nike

我正在使用 Python/Tkinter 构建基于 GUI 的应用程序,该应用程序构建在现有 Python bdb 模块之上。在此应用程序中,我想使控制台中的所有 stdout/stderr 静音并将其重定向到我的 GUI。为了实现这个目的,我编写了一个专门的 Tkinter.Text 对象(代码在文章末尾)。

基本思想是,当向 sys.stdout 写入内容时,它在“文本”中显示为黑色的一行。如果有内容写入 sys.stderr,它会在“文本”中显示为红色的一行。一旦写入内容,文本总是会向下滚动以查看最新的行。

我目前使用的是 Python 2.6.1。在 Mac OS X 10.5 上,这似乎工作得很好。我对它的问题为零。然而,在 RedHat Enterprise Linux 5 上,我很可靠地在脚本运行期间遇到了段错误。段错误并不总是发生在同一个地方,但它几乎总是发生。如果我从我的代码中注释掉 sys.stdout=sys.stderr= 行,段错误似乎就消失了。

我确信我可能不得不采用其他方法来解决这个问题,但是有人能看到我在这里做的任何明显错误的事情可能会导致这些段错误吗?这让我发疯。谢谢!

PS - 我意识到将 sys.stderr 重定向到 GUI 可能不是一个好主意,但即使我只重定向 sys.stdout 而不是 sys.stderr,我仍然会遇到段错误。我还意识到我现在允许文本无限增长。

class ConsoleText(tk.Text):
'''A Tkinter Text widget that provides a scrolling display of console
stderr and stdout.'''

class IORedirector(object):
'''A general class for redirecting I/O to this Text widget.'''
def __init__(self,text_area):
self.text_area = text_area

class StdoutRedirector(IORedirector):
'''A class for redirecting stdout to this Text widget.'''
def write(self,str):
self.text_area.write(str,False)

class StderrRedirector(IORedirector):
'''A class for redirecting stderr to this Text widget.'''
def write(self,str):
self.text_area.write(str,True)

def __init__(self, master=None, cnf={}, **kw):
'''See the __init__ for Tkinter.Text for most of this stuff.'''

tk.Text.__init__(self, master, cnf, **kw)

self.started = False
self.write_lock = threading.Lock()

self.tag_configure('STDOUT',background='white',foreground='black')
self.tag_configure('STDERR',background='white',foreground='red')

self.config(state=tk.DISABLED)

def start(self):

if self.started:
return

self.started = True

self.original_stdout = sys.stdout
self.original_stderr = sys.stderr

stdout_redirector = ConsoleText.StdoutRedirector(self)
stderr_redirector = ConsoleText.StderrRedirector(self)

sys.stdout = stdout_redirector
sys.stderr = stderr_redirector

def stop(self):

if not self.started:
return

self.started = False

sys.stdout = self.original_stdout
sys.stderr = self.original_stderr

def write(self,val,is_stderr=False):

#Fun Fact: The way Tkinter Text objects work is that if they're disabled,
#you can't write into them AT ALL (via the GUI or programatically). Since we want them
#disabled for the user, we have to set them to NORMAL (a.k.a. ENABLED), write to them,
#then set their state back to DISABLED.

self.write_lock.acquire()
self.config(state=tk.NORMAL)

self.insert('end',val,'STDERR' if is_stderr else 'STDOUT')
self.see('end')

self.config(state=tk.DISABLED)
self.write_lock.release()

最佳答案

我假设这是一个更大的线程程序的一部分。

不使用锁,而是将代码写入线程安全的队列对象。然后,在您的主线程中轮询队列并写入文本小部件。您可以使用事件循环进行轮询(而不是编写自己的循环),方法是运行轮询作业,轮询作业会使用 after 重新安排自己在几毫秒后运行(几百毫秒可能就足够了)。

关于python - 将 sys.stdout 重定向到 Tkinter.Text 小部件时出现段错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2914603/

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