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

时间:2017-04-11 11:19:37

标签: windows multithreading wxpython wxwidgets msys2

我正在测试Windows 7,64位,MSYS2 Mingw64 shell(shell启动命令为C:\msys64\msys2_shell.cmd -use-full-path -mingw64);这里我已通过pacman安装:mingw-w64-x86_64-python2-2.7.13-1mingw-w64-x86_64-wxWidgets-3.0.2-17mingw-w64-x86_64-wxPython-3.0.2.0-6

考虑这个代码,它只有标题标签,按钮和目标标签;单击按钮时,标签应从“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下使用线程运行它没有问题)

2 个答案:

答案 0 :(得分:0)

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

这意味着您无法从辅助线程访问GUI元素。仅从主线程支持GUI访问 - 您创建应用程序对象的那个。

答案 1 :(得分:0)

您无法从主要线程以外的任何线程调用影响GUI的方法。而是将事件发布到主线程,要求它代表工作线程执行所需的操作。

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