0

我对 wxpython 的 textctrl 和线程有任何疑问。将不胜感激任何帮助解决这个问题。

我的程序处理文件,当每个文件被处理时,它在 textctrl 中被列为已完成。当只处理几个文件时,textctrl 是响应式的并且会立即显示并且不会消失。即使这些文件很大。对 700mb 文件进行了测试,textctrl 运行良好。

处理许多文件时会出现问题,例如 20+。在这种情况下,textctrl 会消失 6 或 7 秒,然后重新出现并正常工作。

我尝试过普通线程、守护线程等。还尝试使用 .join() ,这让事情变得更糟。我想知道这是否只是因为我的程序是处理器密集型的,或者我只是做错了什么。

下面列出了我的线程代码行。到目前为止,这是迄今为止最快的方法,但对我的目的来说还不够好。在此先感谢克林顿。

def Worker(self, e, _file):

    match = ''

    with open(_file, 'r') as f:
        data = f.read()

    for char in data:
        if char in self.key:
            match += chr(self.key.index(char))

    open(_file, 'w').close()

    with open(_file, 'w') as f:
        f.write(match)

    wx.CallAfter(self.ListFilesEncrypt, e, _file)

if __name__ == '__main__':
    for _file in self.file2process:
        self.filenum += 1
        Thread(target=self.Worker, args=(e, _file,)).start()
4

3 回答 3

3

使用线程安全方法更新 GUI。在 wxPython 中,有 3 个:

  • wx.CallAfter
  • wx.CallLater
  • wx.PostEvent

您还应该查看 wxPython wiki 以获取有关 wxPython 和线程的信息:

我还写了一个关于这个主题的教程:

更新:这是一个简单的例子,它创建了 40 个线程和“进程”40 个组成的文件。当每个线程完成时,它会更新显示。但是,我没有看到你所说的问题。

import random
import time
import wx

from threading import Thread
from wx.lib.pubsub import Publisher

########################################################################
class TestThread(Thread):
    """Test Worker Thread Class."""

    #----------------------------------------------------------------------
    def __init__(self, fname, sleepAmt):
        """Init Worker Thread Class."""
        Thread.__init__(self)
        self.fname = fname
        self.sleepAmt = sleepAmt
        self.start()    # start the thread

    #----------------------------------------------------------------------
    def run(self):
        """Run Worker Thread."""
        # This is the code executing in the new thread.
        time.sleep(self.sleepAmt)
        msg = "%s finished in %s seconds!" % (self.fname, self.sleepAmt)
        wx.CallAfter(Publisher().sendMessage, "update", msg)


########################################################################
class MyForm(wx.Frame):

    #----------------------------------------------------------------------
    def __init__(self):
        wx.Frame.__init__(self, None, wx.ID_ANY, "Tutorial")

        panel = wx.Panel(self, wx.ID_ANY)
        self.updateText = wx.TextCtrl(panel, style=wx.TE_MULTILINE)
        self.btn = btn = wx.Button(panel, label="Start Thread")

        btn.Bind(wx.EVT_BUTTON, self.onButton)

        sizer = wx.BoxSizer(wx.VERTICAL)
        sizer.Add(self.updateText, 1, wx.ALL|wx.EXPAND, 5)
        sizer.Add(btn, 0, wx.ALL|wx.CENTER, 5)
        panel.SetSizer(sizer)

        # create a pubsub receiver
        Publisher().subscribe(self.updateDisplay, "update")

    #----------------------------------------------------------------------
    def onButton(self, event):
        """
        Runs the thread
        """
        for i in range(40):
            fname = "test%s.txt" % i
            secs = random.choice(range(3, 15))
            TestThread(fname, secs)

    #----------------------------------------------------------------------
    def updateDisplay(self, msg):
        """
        Receives data from thread and updates the display
        """
        data = msg.data + "\n"
        self.updateText.WriteText(data)

#----------------------------------------------------------------------
# Run the program
if __name__ == "__main__":
    app = wx.PySimpleApp()
    frame = MyForm().Show()
    app.MainLoop()

我正在使用带有 wxPython 2.8.12.1 的 Python 2.6 在 Windows 7 上运行

于 2013-08-28T17:58:47.190 回答
0

您的线程不应直接更新文本控件-它们应该使用wx.CallAfter或者更好地为主线程(GIU)设置一个标志来更新控件-听起来像结束线程方法可能是合适的。

于 2013-08-28T17:56:58.973 回答
0

除了 CPU 密集型之外,它还可能是 IO 密集型的。IO 密集型应用程序可能会对性能产生非常大的影响,尤其是当您还将 IO 单元用于分页等其他关键目的时。我建议一次服务 5 到 10 份,其余的排队。

于 2013-08-28T18:01:16.333 回答