[英]tkinter progressbar for multiprocessing
我有一個加密文件的程序,我使用了多處理來使其更快,但是我在使用 tkinter 進度條時遇到了問題。
我已經實施了它,但它立即完成或在兩者之間滯后。 進度條剛剛完成到 100%,但文件仍在加密,我沒有收到任何錯誤。
files 包含文件列表。
整個代碼在這里 - https://codeshare.io/pq8YxE
以下是我實施它的方式。
def progbar():
global pb_lable
global percent
global pbar
global percentlabel
global pbar_frame
pb_lable = tk.Label(root, text='Progress', font = "Raleway 13 bold")
pb_lable.grid(row=5, columnspan=2, sticky='w', padx=(35))
pbar_frame = tk.Frame(root)
pbar_frame.grid(row=6, columnspan=2)
pbar = Progressbar(pbar_frame,orient='horizontal',length=500,mode='determinate')
pbar.grid(row=7,column=0, pady=10, padx=20)
percent = tk.StringVar()
percentlabel = tk.Label(root, textvariable=percent, font='Raleway 15')
percentlabel.grid(row=5,columnspan=2,pady=10, padx=120, sticky='w')
def encryptfn(key, a):
f = Fernet(key)
return f.encrypt(a)
def enc(key, process_pool, file):
task = len(files)
x = 0
with open(file,'rb') as original_file:
original = original_file.read()
encrypted = process_pool.apply(encryptfn, args=(key, original,))
with open (file,'wb') as encrypted_file:
encrypted_file.write(encrypted)
pbar['value']+=100/task
x = x+1
percent.set(str(int((x/task)*100))+'%')
root.update_idletasks()
def encfile():
password = bytes('asdasd', 'utf-8')
salt = bytes('zxcasd','utf-8')
global files
files = filistbox.get(0,'end')
if len(files) == 0:
fierrorbox()
elif len(password) == 0:
passerrorbox()
else:
file_enc_button['state']='disabled'
browsefi['state']='disabled'
kdf = PBKDF2HMAC(
algorithm=hashes.SHA256(),
length=32,
salt=salt,
iterations=100,
backend=default_backend())
key = base64.urlsafe_b64encode(kdf.derive(password))
MAX_THREADS = 300
pool_size = min(MAX_THREADS, cpu_count(), len(files))
process_pool = Pool(pool_size)
thread_pool = ThreadPool(min(MAX_THREADS, len(files)))
worker = partial(enc, key, process_pool)
thread_pool.map(worker, files)
root.event_generate("<<encryption_done>>")
file_enc_button['state']='active'
browsefi['state']='active'
def run_encfile():
root.bind('<<encryption_done>>', encryption_done)
Thread(target=encfile).start()
def encryption_done(*args):
fiencdone()
if __name__ == '__main__':
root = tk.Tk()
browsefi = tk.Button(root, text='Browse', command=fibrowse, borderwidth=3)
browsefi.grid(row=2,column=0,padx=5, pady=5)
## File list ##
filist_frame = tk.Frame(root)
filist_frame.grid(row=3, columnspan=2)
filistbox = tk.Listbox(filist_frame, width=40, height=10)
filistbox.grid(row=3,columnspan=2, pady=10)
## Button ##
fibutton_frame = tk.Frame(root)
fibutton_frame.grid(row=4, columnspan=2)
file_enc_button = tk.Button(fibutton_frame, text='Encrypt', width=15, command=run_encfile, borderwidth=3)
file_enc_button.grid(row=4,column=0,padx=10,pady=15)
progbar()
percent.set('0%')
root.mainloop()
問題出現在函數enc
,您不斷將變量x
重置為 0。
您需要一個全局變量,例如tasks_completed
,初始化為 0,它跟蹤已完成的加密數量,並在enc
使用語句tasks_completed += 1
遞增。 從技術上講,這等效於tasks_completed = tasks_completed + 1
並且不是原子操作,並且理論上有可能在讀取tasks_completed
的值后線程可能會被中斷,因此兩個線程會將tasks_completed
更新為相同的值。 所以這個更新,為了完全安全,應該在multithreading.Lock
控制下進行multithreading.Lock
以確保一次只有一個線程可以進行更新。
將函數enc
替換為:
from threading import Lock
tasks_completed = 0
def enc(key, process_pool, lock, file):
global tasks_completed
with open(file,'rb') as original_file:
original = original_file.read()
encrypted = process_pool.apply(encryptfn, args=(key, original,))
with open (file,'wb') as encrypted_file:
encrypted_file.write(encrypted)
encrypted = process_pool.apply(encryptfn, args=(key, file,))
with lock:
tasks_completed += 1
percentage_completed = int((tasks_completed / len(files)) * 100)
pbar['value'] = percentage_completed
percent.set(f'{percentage_completed}%')
root.update_idletasks()
並創建一個multiprocessing.Lock
實例並通過如下修改函數將其傳遞給函數encfile
的enc
:
lock = Lock() # create lock
# add additional lock argument to enc:
worker = partial(enc, key, process_pool, lock)
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.