在MSYS2中的wxPython:线程处理程序中的SetLabel导致冻结?

问题描述:

我在Windows 7,64位,MSYS2 Mingw64壳(壳开始命令是C:\msys64\msys2_shell.cmd -use-full-path -mingw64)测试;这里我已经通过pacman安装:mingw-w64-x86_64-python2-2.7.13-1,mingw-w64-x86_64-wxWidgets-3.0.2-17mingw-w64-x86_64-wxPython-3.0.2.0-6在MSYS2中的wxPython:线程处理程序中的SetLabel导致冻结?

考虑这个代码,其中只有一个标题标签,按钮和目标标签;点击按钮时,标签应改变从“X”为“1”:

import wx #, wx.html 
import sys, os 
from threading import Thread 

# disable buffering (Windows) 
sys.stdout = os.fdopen(sys.stdout.fileno(), 'w', 0) 
sys.stderr = os.fdopen(sys.stderr.fileno(), 'w', 0) 

class Frame(wx.Frame): 
    def __init__(self, *args, **kwds): 
    kwds["style"] = wx.DEFAULT_FRAME_STYLE 
    wx.Frame.__init__(self, *args, **kwds) 

    self.label = wx.StaticText(self, wx.ID_ANY, "Click the button to change label below: ") 
    self.bt_main = wx.Button(self, label="Click ME") 
    self.bt_main.Bind(wx.EVT_BUTTON, self.BtnClickHandler) 
    self.label2 = wx.StaticText(self, wx.ID_ANY, "XX") 

    sizer_vmain_app = wx.BoxSizer(wx.VERTICAL) 
    sizer_vmain_app.Add(self.label, proportion=0, flag=wx.EXPAND, border=0) 
    sizer_vmain_app.Add(self.bt_main, proportion=0, flag=0, border=0) 
    sizer_vmain_app.Add(self.label2, proportion=0, flag=0, border=0) 

    self.SetSizer(sizer_vmain_app) 
    self.Layout() 

    def BtnClickHandler(self, event): 
    testThread = Thread(target=self.DoBtnClick) 
    testThread.start() 
    testThread.join() 

    def DoBtnClick(self): 
    print("BtnClickHandler ") 
    myval = int("1") 
    self.label2.SetLabel(str(myval)) 

if __name__ == "__main__": 
    app = wx.PySimpleApp(0) 
    wx.InitAllImageHandlers() 
    app_frame = Frame(None, wx.ID_ANY, "") 
    app.SetTopWindow(app_frame) 
    app_frame.Show() 
    app.MainLoop() 

当我为这段代码运行,那么当它涉及到self.label2.SetLabel(str(myval))申请冻结。

但是,如果我避免了线程,并使用此功能来代替:

def BtnClickHandler(self, event): 
    # testThread = Thread(target=self.DoBtnClick) 
    # testThread.start() 
    # testThread.join() 
    self.DoBtnClick() 

......然后一切工作正常。请注意,我通过在MSYS2 Mingw64 shell中运行python test.py来调用此脚本。

那么,是不是可以用线程在Windows上运行此代码,如果是这样,怎么样? (否则,在Linux下使用线程运行它没有问题)

+1

你看过wxpython演示吗? – Igor

+0

谢谢@Igor - 不,我没有,它有什么不同?这对我有何帮助? – sdaau

+0

对不起我的坏。请参阅下面的答案。 – Igor

问题是wxWidgets库不是线程安全的。

它的意思是,你不能从辅助线程访问GUI元素。只有主线程支持GUI访问 - 您在其上创建应用程序对象。

不能调用影响来自除主之外的任何线程图形用户界面的方法。相反,将事件发布到主线程,请求它代表工作线程执行所需的操作。

在C++中做到这一点的最简单的方法,到目前为止,是使用与CallAfter()的λ,例如你可以在线程代码中执行CallAfter([=](){ label2->SetLabel(myval); })。不幸的是,我不知道这是否可以从Python获得。