TKINTER TUTORIAL
Chương trình "Hello, World!" bằng Tkinter trong Python Tìm hiểu cửa sổ Tkinter trong Python Tìm hiểu về các Widget Tkinter trong Python 3 cách đặt Options cho Widget Tkinter trong Python Ràng buộc lệnh trong Tkinter Tìm hiểu Event Binding của Tkinter trong Python Giới thiệu widget Label trong Tkinter Cách sử dụng widget Entry trong Tkinter Trình quản lý hình học Pack trong Tkinter Trình quản lý hình học Grid trong Tkinter Trình quản lý hình học Place trong Tkinter Kích thước Widget trong Tkinter Tìm hiểu về widget Frame trong Tkinter Cách sử dụng widget Text của Tkinter Tìm hiểu về widget Scrollbar của Tkinter Cách sử dụng widget ScrolledText của Tkinter Cách sử dụng widget Separator của Tkinter Cách sử dụng Widget Checkbox Trong Tkinter Cách sử dụng widget radio button của Tkinter Cách tạo widget combobox trong Tkinter Cách sử dụng widget Listbox trong Tkinter Sử dụng widget PanedWindow trong Tkinter Cách tạo widget Spinbox trong Tkinter Hướng dẫn sử dụng Slider trong Tkinter Cách sử dụng widget Sizegrip trong Tkinter Cách sử dụng widget LabelFrame trong Tkinter Cách sử dụng widget Progressbar trong Tkinter Cách sử dụng widget Notebook trong Tkinter Cách sử dụng về widget Treeview trong Tkinter Hướng dẫn về Canvas trong Tkinter Cách thiết lập con trỏ widget trong Tkinter. Window hướng đối tượng trong Tkinter Các Frame hướng đối tượng trong Tkinter Cách sử dụng Object-Oriented Application trong Tkinter Phương thức tkraise() của Frame trong Tkinter Các kiểu (Styles) trong Tkinter Cách thay đổi theme trong Tkinter Các element của ttk trong Tkinter Tìm hiểu về đối tượng Tkinter StringVar trong Tkinter Cách sử dụng phương thức map() của ttk.Style Cách sử dụng phương thức after() của Tkinter Cách sử dụng threads trong Tkinter Cách hiển thị progress bar khi thread đang chạy trong Tkinter. Cách tạo nhiều cửa sổ trong một Tkinter Cách sử dụng widget PhotoImage của Tkinter Cấu trúc MVC trong Tkinter Cách sử dụng Validate trong Tkinter Cách hiển thị đồ thị từ thư viện Matplotlib trong Tkinter Ứng dụng System Tray với Tkinter
CÁC CHỦ ĐỀ
BÀI MỚI NHẤT
MỚI CẬP NHẬT

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.

test php

banquyen png
Bài viết này được đăng tại freetuts.net, không được copy dưới mọi hình thức.

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:

Tkinter Thread Example png

Để 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ớp AsyncDownload, chúng ta khởi tạo các thuộc tính htmlurl.
  • Trong phương thức run(), chúng ta gọi hàm get() để tải trang web được chỉ định bởi URL và gán mã nguồn HTML cho thuộc tính html.

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ùng chuyên mục:

Cách thêm Progress Bar trong Python với chỉ một dòng Code

Cách thêm Progress Bar trong Python với chỉ một dòng Code

Toán tử Walrus Operator- Tính năng mới trong Python 3.8

Toán tử Walrus Operator- Tính năng mới trong Python 3.8

Cách nạp dữ liệu Machine Learning từ File trong Python

Cách nạp dữ liệu Machine Learning từ File trong Python

Hướng dẫn sử dụng Google Sheets API với Python

Hướng dẫn sử dụng Google Sheets API với Python

Xây dựng  web Python tự động hóa Twitter | Flask, Heroku, Twitter API & Google Sheets API

Xây dựng web Python tự động hóa Twitter | Flask, Heroku, Twitter API & Google Sheets API

Xây dựng Web Machine Learning đẹp mắt với Streamlit và Scikit-learn trong Python

Xây dựng Web Machine Learning đẹp mắt với Streamlit và Scikit-learn trong Python

Hướng dẫn tạo Chatbot đơn giản bằng PyTorch

Hướng dẫn tạo Chatbot đơn giản bằng PyTorch

11 mẹo và thủ thuật để viết Code Python hiệu quả hơn

11 mẹo và thủ thuật để viết Code Python hiệu quả hơn

Hướng dẫn làm ứng dụng TODO với Flask dành cho người mới bắt đầu trong Python

Hướng dẫn làm ứng dụng TODO với Flask dành cho người mới bắt đầu trong Python

Hướng dẫn viết Snake Game bằng Python

Hướng dẫn viết Snake Game bằng Python

Cách sử dụng chế độ interactive trong Python

Cách sử dụng chế độ interactive trong Python

Cách sử dụng Python Debugger với hàm breakpoint()

Cách sử dụng Python Debugger với hàm breakpoint()

Xây dựng ứng dụng Web Style Transfer với PyTorch và Streamlit

Xây dựng ứng dụng Web Style Transfer với PyTorch và Streamlit

Cách cài đặt Jupyter Notebook trong môi trường Conda và thêm Kernel

Cách cài đặt Jupyter Notebook trong môi trường Conda và thêm Kernel

Hướng dẫn xây dựng ứng dụng dự đoán giá cổ phiếu bằng Python

Hướng dẫn xây dựng ứng dụng dự đoán giá cổ phiếu bằng Python

Hướng dẫn tạo ứng dụng AI hội thoại với NVIDIA Jarvis trong Python

Hướng dẫn tạo ứng dụng AI hội thoại với NVIDIA Jarvis trong Python

Hỗ trợ Async trong Django 3.1

Hỗ trợ Async trong Django 3.1

8 mẹo tái cấu trúc Python giúp mã sạch hơn và Pythonic

8 mẹo tái cấu trúc Python giúp mã sạch hơn và Pythonic

Ý nghĩa của if __name__ ==

Ý nghĩa của if __name__ == "__main__" trong Python

Cách xóa phần tử trong danh sách Python

Cách xóa phần tử trong danh sách Python

Top