gpt4 book ai didi

python - 从 Tkinter Tcl 回调到 python 函数在 Windows 中崩溃

转载 作者:行者123 更新时间:2023-12-05 09:20:10 24 4
gpt4 key购买 nike

这不完全是我的应用程序,但非常相似。我创建了这个测试代码来显示问题。基本上我试图从 python 线程调用 tcl proc。当结果准备好时,Tcl proc 将回调到 python 函数。此结果将作为事件发布到 wx 框架。当我作为纯 python 代码运行时,它工作正常。当我使用 tcl proc 时,整个应用程序在没有任何信息的情况下崩溃。如果我增加 wait_time(比如 100),那么即使使用 tcl 也能正常工作。是高回拨率问题还是我遗漏了其他问题。顺便说一句,这个应用程序在 Windows 上运行。

import wx
from Tkinter import Tcl
from threading import Thread
import wx.lib.newevent
from time import sleep

CountUpdateEvent, EVT_CNT_UPDATE = wx.lib.newevent.NewEvent()

tcl_code = 'proc tcl_worker {name max_count delay_time callback} { while {$max_count>0} {after $delay_time; $callback $name $max_count; incr max_count -1}}'

# Option to use Tcl or not for counter
# When enabled, Tcl will callback to python to upate counter value
use_tcl = True

# Option to create tcl interpreter per thread.
# Test shows single interpreter for all threads will fail.
use_per_thread_tcl = True

count = 5000
wait_time = 1 ;# in milliseconds

class Worker:
def __init__(self,name,ui,tcl):
global use_per_thread_tcl
self.name = name
self.ui = ui
if use_per_thread_tcl:
self.tcl = Tcl()
self.tcl.eval(tcl_code)
else:
self.tcl = tcl
self.target = ui.add_textbox(name)
self.thread = Thread(target=self.run)
self.thread.daemon = True
self.thread.start()

def callback(self, name, val):
evt = CountUpdateEvent(name=self.name, val=val, target=self.target)
wx.PostEvent(self.ui,evt)
def run(self):
global count, wait_time, use_tcl

if use_tcl:
# Register a python function to be called back from tcl
tcl_cmd = self.tcl.register(self.callback)

# Now call tcl proc
self.tcl.call('tcl_worker', self.name, str(count), str(wait_time), tcl_cmd)
else:
# Convert milliseconds to seconds for sleep
py_wait_time = wait_time / 1000
while count > 0:
# Directly call the callback from here
self.callback(self.name, str(count))
count -= 1
sleep(py_wait_time)


class MainWindow(wx.Frame):
def __init__(self, parent):
wx.Frame.__init__(self, parent, title="Decrement Counter", size=(600, 100))

self._DoLayout()
self.Bind(EVT_CNT_UPDATE, self.on_count_update)

def _DoLayout(self):
self.sizer = wx.BoxSizer(wx.HORIZONTAL)

self.panels = []
self.tbs = []
self.xpos = 0

def add_textbox(self,name):
panel = wx.Panel(self, pos=(self.xpos, 0), size=(60,40))
self.panels.append(panel)
tb = wx.StaticText(panel, label=name)
tb.SetFont(wx.Font(16,wx.MODERN,wx.NORMAL,wx.NORMAL))
self.sizer.Add(panel, 1, wx.EXPAND, 7)
self.tbs.append(tb)
self.xpos = self.xpos + 70
return tb

def on_count_update(self,ev):
ev.target.SetLabel(ev.val)
del ev

if __name__ == '__main__':
app = wx.App(False)
frame = MainWindow(None)
tcl = Tcl()
tcl.eval(tcl_code)
w1 = Worker('A', frame, tcl)
w2 = Worker('B', frame, tcl)
w3 = Worker('C', frame, tcl)
w4 = Worker('D', frame, tcl)
w5 = Worker('E', frame, tcl)
w6 = Worker('F', frame, tcl)
w7 = Worker('G', frame, tcl)
w8 = Worker('H', frame, tcl)
frame.Show()
app.MainLoop()

最佳答案

每个 Tcl 解释器对象(即知道如何运行 Tcl 过程的上下文)只能从创建它的 OS 线程安全地使用。这是因为 Tcl 没有像 Python 那样使用全局解释器锁,而是大量使用线程特定的数据来减少内部所需的锁的数量。 (编写良好的 Tcl 代码可以利用这一点在合适的硬件上扩展得非常大。)

因此,您必须确保您只从单个线程运行 Tcl 命令或 Tkinter 操作;这通常是主线程,但我不确定这是否是与 Python 集成的真正要求。如果你愿意,你可以启动工作线程,但它们将无法使用 Tcl 或 Tkinter(好吧,不是没有非常特殊的预防措施,这些预防措施比它可能值得的麻烦更多)。相反,他们需要向主线程发送消息,以便它处理与 GUI 的交互;有许多不同的方法可以做到这一点。

关于python - 从 Tkinter Tcl 回调到 python 函数在 Windows 中崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38767355/

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