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ínhhtmlvà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.

Các kiểu dữ liệu trong C ( int - float - double - char ...)
Thuật toán tìm ước chung lớn nhất trong C/C++
Cấu trúc lệnh switch case trong C++ (có bài tập thực hành)
ComboBox - ListBox trong lập trình C# winforms
Random trong Python: Tạo số random ngẫu nhiên
Lệnh cin và cout trong C++
Cách khai báo biến trong PHP, các loại biến thường gặp
Download và cài đặt Vertrigo Server
Thẻ li trong HTML
Thẻ article trong HTML5
Cấu trúc HTML5: Cách tạo template HTML5 đầu tiên
Cách dùng thẻ img trong HTML và các thuộc tính của img
Thẻ a trong HTML và các thuộc tính của thẻ a thường dùng