Thông báo: Download 4 khóa học Python từ cơ bản đến nâng cao tại đây.
Cách sử dụng threads trong Tkinter
Trong quá trình phát triển giao diện người dùng (GUI) với Tkinter, việc duy trì khả năng phản hồi của ứng dụng là rất quan trọng, đặc biệt khi bạn phải xử lý các tác vụ nền phức tạp như tải dữ liệu từ internet, xử lý tệp tin lớn, hoặc thực hiện các phép tính phức tạp. Để giải quyết vấn đề này, bạn có thể sử dụng luồng (threads) trong Tkinter. Trong bài viết này, bạn sẽ học cách tích hợp nhiều luồng vào ứng dụng Tkinter của mình để giúp nó hoạt động mượt mà và phản hồi nhanh hơn ngay cả khi thực hiện các tác vụ nặng.
Khi nào nên sử dụng Thread trong ứng dụng Tkinter
Trong một ứng dụng Tkinter, vòng lặp chính (main loop) luôn phải được khởi động trong luồng chính. Nó chịu trách nhiệm xử lý các sự kiện và cập nhật giao diện người dùng (GUI).
Nếu bạn có một thao tác nền (background operation) mất nhiều thời gian, bạn nên thực hiện nó trong một luồng riêng biệt.
Nếu không, ứng dụng sẽ không phản hồi và thậm chí có thể bị treo khi thao tác đó đang chạy.
Bài viết này được đăng tại [free tuts .net]
Để tạo và điều khiển nhiều luồng trong ứng dụng Tkinter, bạn có thể sử dụng mô-đun threading
của Python.
Mô-đun threading
được bao gồm trong thư viện tiêu chuẩn của Python, vì vậy bạn không cần phải cài đặt nó.
Ví dụ về sử dụng Thread trong Tkinter
Mình sẽ xây dựng một chương trình đơn giản để tải về nội dung của một trang web được chỉ định bởi URL và hiển thị nội dung đó trong một widget Text:
Để tải xuống một trang web, mình sẽ sử dụng mô-đun requests
.
Bước 1: Cài đặt mô-đun requests
bằng lệnh
pip install requests
Bước 2: Nhập các mô-đun cần thiết: tkinter
, threading
, và requests
import tkinter as tk from tkinter import ttk from tkinter.messagebox import showerror from threading import Thread import requests
Bước 3: Định nghĩa một lớp mới gọi là AsyncDownload
kế thừa từ lớp Thread
class AsyncDownload(Thread): def __init__(self, url): super().__init__() self.html = None self.url = url def run(self): response = requests.get(self.url) self.html = response.text
Cách hoạt động của lớp AsyncDownload
:
- Trong phương thức
__init__()
của lớpAsyncDownload
, chúng ta khởi tạo các thuộc tínhhtml
vàurl
. - Trong phương thức
run()
, chúng ta gọi hàmget()
để tải trang web được chỉ định bởi URL và gán mã nguồn HTML cho thuộc tínhhtml
.
Bước 4: Tạo lớp App
kế thừa từ lớp Tk
. Lớp App
đại diện cho cửa sổ gốc.
Cửa sổ gốc bao gồm ba khung (frame) chứa tất cả các widget. Khi bạn nhấn nút tải xuống, chương trình sẽ thực hiện phương thức handle_download()
của lớp App
.
Trong phương thức handle_download()
, chúng ta kiểm tra xem URL có được cung cấp không. Nếu có, chúng ta tạo một đối tượng mới của lớp AsyncDownload
và khởi động luồng. Đồng thời, chúng ta vô hiệu hóa nút tải xuống và xóa nội dung của widget Text.
Ngoài ra, chúng ta gọi phương thức monitor()
để theo dõi trạng thái của luồng.
def handle_download(self): url = self.url_var.get() if url: self.download_button['state'] = tk.DISABLED self.html.delete(1.0, "end") download_thread = AsyncDownload(url) download_thread.start() self.monitor(download_thread) else: showerror(title='Error', message='Vui lòng nhập URL của trang web.')
Bước 5: Trong phương thức monitor()
, chúng ta lên lịch thực hiện một hành động sẽ gọi lại phương thức monitor()
sau mỗi 100ms nếu luồng vẫn đang chạy.
Nếu việc tải xuống hoàn tất, chúng ta cập nhật nội dung cho widget Text và bật lại nút tải xuống:
def monitor(self, thread): if thread.is_alive(): self.after(100, lambda: self.monitor(thread)) else: self.html.insert(1.0, thread.html) self.download_button['state'] = tk.NORMAL
Bước 6: Cuối cùng, chạy vòng lặp chính của ứng dụng:
if __name__ == "__main__": app = App() app.mainloop()
Toàn bộ chương trình như sau:
import tkinter as tk from tkinter import ttk from tkinter.messagebox import showerror from threading import Thread import requests class AsyncDownload(Thread): def __init__(self, url): super().__init__() self.html = None self.url = url def run(self): response = requests.get(self.url) self.html = response.text class App(tk.Tk): def __init__(self): super().__init__() self.title('Webpage Download') self.geometry('680x430') self.resizable(0, 0) self.create_header_frame() self.create_body_frame() self.create_footer_frame() def create_header_frame(self): self.header = ttk.Frame(self) self.header.columnconfigure(0, weight=1) self.header.columnconfigure(1, weight=10) self.header.columnconfigure(2, weight=1) self.label = ttk.Label(self.header, text='URL') self.label.grid(column=0, row=0, sticky=tk.W) self.url_var = tk.StringVar() self.url_entry = ttk.Entry(self.header, textvariable=self.url_var, width=80) self.url_entry.grid(column=1, row=0, sticky=tk.EW) self.download_button = ttk.Button(self.header, text='Download') self.download_button['command'] = self.handle_download self.download_button.grid(column=2, row=0, sticky=tk.E) self.header.grid(column=0, row=0, sticky=tk.NSEW, padx=10, pady=10) def handle_download(self): url = self.url_var.get() if url: self.download_button['state'] = tk.DISABLED self.html.delete(1.0, "end") download_thread = AsyncDownload(url) download_thread.start() self.monitor(download_thread) else: showerror(title='Error', message='Vui lòng nhập URL của trang web.') def monitor(self, thread): if thread.is_alive(): self.after(100, lambda: self.monitor(thread)) else: self.html.insert(1.0, thread.html) self.download_button['state'] = tk.NORMAL def create_body_frame(self): self.body = ttk.Frame(self) self.html = tk.Text(self.body, height=20) self.html.grid(column=0, row=1) scrollbar = ttk.Scrollbar(self.body, orient='vertical', command=self.html.yview) scrollbar.grid(column=1, row=1, sticky=tk.NS) self.html['yscrollcommand'] = scrollbar.set self.body.grid(column=0, row=1, sticky=tk.NSEW, padx=10, pady=10) def create_footer_frame(self): self.footer = ttk.Frame(self) self.footer.columnconfigure(0, weight=1) self.exit_button = ttk.Button(self.footer, text='Exit', command=self.destroy) self.exit_button.grid(column=0, row=0, sticky=tk.E) self.footer.grid(column=0, row=2, sticky=tk.NSEW, padx=10, pady=10) if __name__ == "__main__": app = App() app.mainloop()
Kết bài
Bằng cách sử dụng các luồng (threads) trong ứng dụng Tkinter, bạn có thể đảm bảo rằng giao diện người dùng luôn mượt mà và phản hồi nhanh ngay cả khi thực hiện các tác vụ nền phức tạp. Việc tách biệt các tác vụ nặng ra khỏi luồng chính giúp tránh tình trạng ứng dụng bị "đơ" hoặc không phản hồi, mang lại trải nghiệm tốt hơn cho người dùng. Như vậy, áp dụng đúng cách các luồng trong Tkinter không chỉ giúp tối ưu hóa hiệu suất mà còn nâng cao chất lượng của ứng dụng bạn phát triển.