前段時間我編寫了一個工業控制的軟體,在使用中一直存在一個問題,就是當軟體檢索裝置時,因為這個功能執行的時間比較長,導致GUI介面假死,讓使用者分辨不清楚軟體到底仍在執行,還是真的掛掉了。(雖然我設計了同步log顯示,但是這個也同樣假死了)
程式如下:
代碼解析如下:
# -*- coding: utf-8 -*- import timeimport wxfrom threading import Threadfrom wx.lib.pubsub import Publisher
time庫用來執行定時函數,類比需要長時間執行的功能。Publisher用來線上程間傳輸訊息。
class TestThread(Thread): def __init__(self): #線程執行個體化時立即啟動 Thread.__init__(self) self.start() def run(self): #線程執行的代碼 for i in range(101): time.sleep(0.03) wx.CallAfter(Publisher().sendMessage, "update", i) time.sleep(0.5) wx.CallAfter(Publisher().sendMessage, "update", u"線程結束")
TestThread的__init__函數,定義線程執行個體化時就啟動,run函數就是需要長時間執行的程式。
wx.CallAfter與Publisher().sendMessage用來向GUI發送訊息。
class MyForm ( wx.Frame ): def __init__( self, parent ): wx.Frame.__init__ ( self, parent, id = wx.ID_ANY, title = u"Leniy,20140627", pos = wx.DefaultPosition, size = wx.Size( -1,-1 ), style = wx.DEFAULT_FRAME_STYLE|wx.TAB_TRAVERSAL ) self.SetSizeHintsSz( wx.DefaultSize, wx.DefaultSize ) gSizer2 = wx.GridSizer( 0, 3, 0, 0 ) self.m_button2 = wx.Button( self, wx.ID_ANY, u"執行線程", wx.DefaultPosition, wx.DefaultSize, 0 ) gSizer2.Add( self.m_button2, 0, wx.ALL|wx.ALIGN_CENTER_HORIZONTAL|wx.ALIGN_CENTER_VERTICAL, 5 ) self.m_staticText2 = wx.StaticText( self, wx.ID_ANY, u"MyLabel", wx.DefaultPosition, wx.DefaultSize, 0 ) self.m_staticText2.Wrap( -1 ) gSizer2.Add( self.m_staticText2, 0, wx.ALL|wx.ALIGN_CENTER_HORIZONTAL|wx.ALIGN_CENTER_VERTICAL, 5 ) self.m_gauge1 = wx.Gauge( self, wx.ID_ANY, 100, wx.DefaultPosition, wx.DefaultSize, wx.GA_HORIZONTAL ) self.m_gauge1.SetValue( 0 ) gSizer2.Add( self.m_gauge1, 0, wx.ALL|wx.ALIGN_CENTER_HORIZONTAL|wx.ALIGN_CENTER_VERTICAL, 5 ) self.SetSizer( gSizer2 ) self.Layout() gSizer2.Fit( self ) self.Centre( wx.BOTH ) self.m_button2.Bind( wx.EVT_BUTTON, self.onButton ) Publisher().subscribe(self.updateDisplay, "update") def updateDisplay(self, msg): t = msg.data if isinstance(t, int):#如果是數字,說明線程正在執行,顯示數字 self.m_staticText2.SetLabel("%s%%" % t) self.m_gauge1.SetValue( t ) else:#否則線程未執行,將按鈕重新開啟 self.m_staticText2.SetLabel("%s" % t) self.m_button2.Enable() def onButton( self, event ): TestThread() self.m_staticText2.SetLabel(u"線程開始") event.GetEventObject().Disable()
這個就是GUI的主程式,__init__中使用Publisher().subscribe(self.updateDisplay, “update”)聲明了擷取線程訊息以及顯示的方法。
if __name__ == "__main__": app = wx.PySimpleApp() MyForm(None).Show() app.MainLoop()
最後將視窗MyForm(None).Show()顯示出來,程式就完成了。