1

编辑:答案如下。 self.progress.update_idletasks()是更新进度条的秘密握手。谢谢麦吉尔森。

我目前正在处理一些较大的 csv 文件(有时高达 700Mb),这需要一段时间(从网络驱动器中提取)。我希望能够给用户一些反馈作为过程的状态。这导致我进入 ttk 状态栏。我查看了尽可能多的示例,然后尝试将它们应用到我的程序中,但没有成功。我认为我的问题是我正在使用for循环来读取我的 csv 文件,但这可能不是问题。我不知道。下面是我的代码如何工作的示例。

import csv
import Tkinter as tk
import ttk

class MAIN(tk.Frame):
    def __init__(self, master, *args, **kwargs):
        n=0
        self.filename = 'some.csv'
        tk.Frame.__init__(self, *args, **kwargs)
        frame=tk.Frame(master, **kwargs)
        frame.grid()
        getdata = tk.Button(frame, text='Get Data', command=self.csvget)
        getdata.grid(row=n)
        n+=1
        self.progress = ttk.Progressbar(frame, orient='horizontal',
                                        length =200, mode='determinate')
        self.progress.grid(row=n)
    def prog(self):
        self.progress['value']=0
        try: f=open(self.filename, 'rb')
        except: return
        read = csv.reader(f, delimiter=',')
        self.maxrows=0
        for row in read: self.maxrows+=1
        self.progress['maximum']=self.maxrows
    def csvget(self):
        try: f=open(self.filename, 'rb')
        except: return
        read = csv.reader(f, delimiter=',')
        q=0
        for row in read:
            if q%100==0: self.progress['value']=q
            elif q==self.maxrows: self.progress['value']=q
            ... do some stuff (read into lists etc.)...

root=tk.Tk()
app=MAIN(root)
app.mainloop()

好吧,结果比我预期的要长。我知道我正在阅读该文件两次。(我无法弄清楚如何以任何其他方式获取 csv 文件的长度,如果您对此有解决方案,我会全力以赴。)但我的问题的主要重点仍然是如何获得进度条工作。在其当前配置中,一旦文件完成读取,它就会立即从零变为满。建议?

编辑:在 mgilson 的帮助下,我的代码变得更好了。但是我的状态栏在文件时仍然没有更新。我尝试了以下方法:

self.completed = tk.IntVar()
self.completed.set(0)
self.progress = ttk.Progressbar(self.frame, orient='horizontal', length=200,
                   mode='determinate', maximum=100, variable=self.completed)
#in the function now
fsize = os.path.getsize(self.filename)
for r,row in enumerate(read):
    q=float(f.tell())/float(fsize)*100 
    if r%100 ==0:
        self.completed.set(q)

另外,我不只是调用,self.completed而是将 adef作为q输入,然后将其放入self.completed. 另外,self.completed我没有使用variablekwarg in self.progress,而是直接和通过def. 也尝试self.progress.configure(value = q)过,它的行为与其他人一样。

所有这些尝试仅在文件完成后刷新栏。

for如果我在读取数据的循环之前修改栏的状态,但在读取def文件完成之前,栏不会更新。(我尝试将其设置为 50 作为实验,文件完成后它从无变为 100%。)

4

1 回答 1

1

你想要类似的东西:

if q%100==0: 
    self.progress['value']=float(q)/100
    self.progress.update_idletasks()

也许:

self.progress.configure(value=q)
self.progress.update_idletasks()

在之前的版本中,只要你读到 100 行,进度条就会被填满。

另请注意,只有在程序控制返回主循环后,进度条才会更新,除非您明确告知它(使用update_idletasks())。

只要不两次读取文件,您可以使用file.tell它(不保证 - 如果它不起作用,请告诉我,我将删除此建议):

fsize=os.path.getsize(filename)
...
q=f.tell()/fsize

如果这可行,您可以将构造函数更改为:

self.progress = ttk.Progressbar(frame, orient='horizontal',
                                length =200, mode='determinate',maximum=1.)

和你的循环:

for i,row in enumerate(read):
    q=f.tell()/fsize
    if i%100==0: 
        self.progress['value']=q
        self.progress.update_idletasks()
    #do stuff
else:
    self.progress['value']=q
    self.progress.update_idletasks()

另一个侧面说明,有一个简单的try/except条款通常不是一个好主意。你通常想要:

try:
   #something
except SomeError: #IOError?  OSError? some combination of the two of them?
   #something else.

仅捕获您知道如何处理的异常。

于 2012-07-02T15:20:21.487 回答