Python tkinter:在子进程调用之间更新GUI

时间:2011-10-18 00:28:38

标签: python tkinter subprocess

我编写了一个多次调用.cmd文件(使用不同参数)的GUI

class App:
    def process(self):
        for filename in os.listdir(path):
            subprocess.call(['script.cmd', filename])
            self.output('processed ' + filename)

    def output(self, line):
        self.textarea.config(state = NORMAL)
        self.textarea.tag_config("green", background="green", foreground="black")
        self.textarea.insert(END, line, ("green"))
        self.textarea.yview(END)
        self.textarea.config(state = DISABLED)
        self.textarea.update_idletasks()

root = Tk()
app = App()
app.build_gui(root)
app.pack_gui(root)

root.mainloop()

按下按钮时调用process()

我也尝试过subprocess.Popen()和旧的os.spawnv() 它总是一样的。 GUI在处理文件时没有反应。只有在处理完所有文件后,才会使用所有“已处理的XYZ”消息更新GUI。

每次子进程调用后,update_idletasks()都不应更新GUI吗?

谢谢

编辑: 我把问题缩小到这个简单的代码:

from Tkinter import *
import subprocess

file_list = ['file1', 'file2', 'file3', 'file4', 'file5']

def go():
    labeltext.set('los')
    for filename in file_list:
        labeltext.set('processing ' + filename + '...')
        label.update_idletasks()

        proc = subprocess.call(["C:\\test\\process.exe", filename])
    labeltext.set('all done!')


root = Tk()

Button(root, text="Go!", command=go).pack(side=TOP)

labeltext = StringVar()
labeltext.set('Press button to start')

label = Label(root, textvariable=labeltext)
label.pack(side=TOP)

root.mainloop()

现在,如果脚本正常工作,它依赖于process.exe。如果我用繁忙循环编写一个简单的C程序(例如process.exe的源代码:int i = 0; while(i< 1e9){i ++;}),则用每个文件1-5更新GUI。当我调用我想要使用的原始.exe文件时,它会显示“处理文件1”并切换到“处理文件2”,然后冻结直到程序终止(“全部完成!”)。

我真的不明白这里有什么。显然它与所谓的过程有关。有没有人有想法?

1 个答案:

答案 0 :(得分:4)

我找到了一个肮脏的解决方案: 我在每个subprocess.call()之前调用root.update()。

为了确保在处理期间没有按下任何按钮(根据快速谷歌搜索,这似乎是root.update()的问题),我在子进程启动之前禁用所有按钮

像这样:

from Tkinter import *
import subprocess

file_list = ['file1', 'file2', 'file3', 'file4', 'file5']

def button():
    b_process.configure(state=DISABLED)
    go()
    b_process.configure(state=NORMAL)

def go():
    for filename in file_list:
        label.configure(text="processing " + filename)
        root.update()

        proc = subprocess.call(["C:\\DTNA\\stat\\run.exe", filename])
        print 'process terminated with return code ' + str(proc)     
    label.configure(text="all done!")

root = Tk()

b_process = Button(root, text="Go!", command=button)
b_process.pack(side=TOP)

label = Label(root, text='Press button to start')
label.pack(side=TOP)

root.mainloop()