Tkinter 面向对象框架《二》

一、说明

        Tkinter 教程 » 开发完整的 Tkinter 面向对象应用程序开发完整的 Tkinter 面向对象应用程序。

        即使OOP的高手,也未必对面向对象全部掌握。至于 Tkinter的OOP编程,其实高手们也是在摸索实践中。 为了面向对象和Tkinter参与本教程。如果你来这里纯粹是为了学习面向对象编程,那很好。

二、第一个实验演示

        在本教程中,您将学习如何开发一个完整的 Tkinter 面向对象的应用程序。您需要将温度转换器应用程序转换为使用面向对象编程方法的新应用程序:

        首先,定义一个名为TemperatureConverter .该类有一个静态方法,用于将温度从华氏度转换为摄氏度:

import tkinter as tk
from tkinter import ttk
from tkinter.messagebox import showerror

class TemperatureConverter:
    @staticmethod
    def fahrenheit_to_celsius(f):
        return (f - 32) * 5 / 9

        其次,定义一个继承自 ttk.Frame 类的 ConverterFrame 类。 ConverterFrame 类将负责创建小部件和处理事件:


class ConverterFrame(ttk.Frame):
    def __init__(self, container):
        super().__init__(container)
        # field options
        options = {'padx': 5, 'pady': 5}

        # temperature label
        self.temperature_label = ttk.Label(self, text='Fahrenheit')
        self.temperature_label.grid(column=0, row=0, sticky=tk.W, **options)

        # temperature entry
        self.temperature = tk.StringVar()
        self.temperature_entry = ttk.Entry(self, textvariable=self.temperature)
        self.temperature_entry.grid(column=1, row=0, **options)
        self.temperature_entry.focus()

        self.convert_button = ttk.Button(self, text='Convert')
        self.convert_button['command'] = self.convert
        self.convert_button.grid(column=2, row=0, sticky=tk.W, **options)

        # result label
        self.result_label = ttk.Label(self)
        self.result_label.grid(row=1, columnspan=3, **options)

        # add padding to the frame and show it
        self.grid(padx=10, pady=10, sticky=tk.NSEW)

    def convert(self):
        """  Handle button click event
        """
        try:
            f = float(self.temperature.get())
            c = TemperatureConverter.fahrenheit_to_celsius(f)
            result = f'{f} Fahrenheit = {c:.2f} Celsius'
            self.result_label.config(text=result)
        except ValueError as error:
            showerror(title='Error', message=error)

工作原理:

  • 需要一个容器,因此,它的方法有参数。ConverterFrame__init__()container
  • 在类的方法中,调用其超类的方法。__init__()ConverterCFrame__init__()
  • 将小部件分配给对象,以便可以在类的其他方法中引用它们。selfConverterFrame
  • 将按钮的选项分配给方法。commandconvertself.convert

第三,定义一个继承自该类的类:Apptk.Tk

class App(tk.Tk):
    def __init__(self):
        super().__init__()

        self.title('Temperature Converter')
        self.geometry('300x70')
        self.resizable(False, False)

最后,从块引导应用程序:if __name__ == "__main__"

if __name__ == "__main__":
    app = App()
    ConverterFrame(app)
    app.mainloop()

把它们放在一起:

import tkinter as tk
from tkinter import ttk
from tkinter.messagebox import showerror


class TemperatureConverter:
    @staticmethod
    def fahrenheit_to_celsius(f):
        return (f - 32) * 5 / 9


class ConverterFrame(ttk.Frame):
    def __init__(self, container):
        super().__init__(container)
        # field options
        options = {'padx': 5, 'pady': 5}

        # temperature label
        self.temperature_label = ttk.Label(self, text='Fahrenheit')
        self.temperature_label.grid(column=0, row=0, sticky=tk.W, **options)

        # temperature entry
        self.temperature = tk.StringVar()
        self.temperature_entry = ttk.Entry(self, textvariable=self.temperature)
        self.temperature_entry.grid(column=1, row=0, **options)
        self.temperature_entry.focus()

        self.convert_button = ttk.Button(self, text='Convert')
        self.convert_button['command'] = self.convert
        self.convert_button.grid(column=2, row=0, sticky=tk.W, **options)

        # result label
        self.result_label = ttk.Label(self)
        self.result_label.grid(row=1, columnspan=3, **options)

        # add padding to the frame and show it
        self.grid(padx=10, pady=10, sticky=tk.NSEW)

    def convert(self):
        """  Handle button click event
        """
        try:
            f = float(self.temperature.get())
            c = TemperatureConverter.fahrenheit_to_celsius(f)
            result = f'{f} Fahrenheit = {c:.2f} Celsius'
            self.result_label.config(text=result)
        except ValueError as error:
            showerror(title='Error', message=error)


class App(tk.Tk):
    def __init__(self):
        super().__init__()

        self.title('Temperature Converter')
        self.geometry('300x70')
        self.resizable(False, False)


if __name__ == "__main__":
    app = App()
    ConverterFrame(app)
    app.mainloop()

在本教程中,您学习了如何开发一个完整的面向对象的 Tkinter 应用程序。

三、Tkinter 面向对象编程速成班

3.1 使用OOP设计Tlinter的多窗口应用

        对于那些不熟悉或困惑于面向对象编程的人来说,你并不孤单。即使是使用它的人有时通常也无法完全了解其内部工作原理。我不认为自己是 OOP 专家,特别是因为我很少使用它,但我知道它足以知道从长远来看,它将帮助我们的 Tkinter 应用程序,我可以为你们分享我所知道的好人!

        因此,面向对象编程是一种编程范式,或者更确切地说:一种结构。就是这样。它只是我们用来构建程序的结构。Python 通常被视为一种纯粹的脚本语言,但实际上它从根本上说是一种 OOP 语言。

        在 OOP 中,你基本上陈述了程序的结构,而你的类从字面上返回“对象”,这就是为什么它被称为面向“对象”的原因。这些对象充当类的“实例”。在我们举个例子之前,我想说的就是这些。我认为一个实际的例子对帮助学习有很长的路要走,所以让我们开始吧!

        我将分块分享代码,解释每个步骤。如果你迷路了,我会在最底部发布这个系列代码的“完整”版本,所以不要害怕!

import tkinter as tk

class SeaofBTCapp(tk.Tk):
		

        我们从简单地将 tkinter 导入为 tk 开始。如果您使用的是 Python 2,请记住 tkinter 称为 Tkinter。

        之后,我们定义我们的SeaofBTCapp类,并传递tk。Tk 看起来就是我们所知道的参数。首先,类根本不需要任何括号。我可以做类SeaofBTCapp:这不会是语法错误。那么什么是tk。那么Tk呢?当你在括号中看到这样的内容时,这意味着该类是从另一个类继承而来的。在我们的例子中,我们继承了 tk 的所有内容。Tk 类。把它想象成你如何导入模块来使用它们。这基本上就是你继承时发生的事情,只是在本地类级别。

        现在,在我们的class中,我们有:

    def __init__(self, *args, **kwargs):
		

        虽然不是必需的,但您通常会将类中的第一个“函数”视为__init__。首先,这些不是函数,即使它们的行为就像函数一样。它们实际上被称为“方法”,__init__是一种非常特殊的方法,因为这种方法总是运行的。Init 缺少初始化,并且只要调用该类,您在此处放入的任何内容都将始终运行。其他方法仅在您专门调用它们运行时才会运行。可以将其视为启动时在计算机上运行的各种启动过程。您希望某些事情始终在计算机打开时启动。您希望鼠标驱动程序联机,需要键盘工作,希望图形驱动程序将桌面抽出,等等。您可能拥有的其他程序,您只希望它们在您单击其图标时启动。这些方法与其他方法类似。

        现在我们已经知道了,我们看到 “init” 中的第一个参数是 “self”。这纯粹是出于标准,实际上不是必需的。你不需要它,你可以完全称它为别的东西,比如“墨西哥卷饼”。称其为“自我”是个好主意,因为这是公认的做法。但是,如果您的目标是混淆,则可以重命名它。因此,self 只是所有类方法的第一个参数。然后你看,我们称它们为“*args”和“**kwargs”。

        就像“self”一样,实际上输入“args”和“kwargs”是没有必要的,这是技巧的星号。添加“args”和“kwargs”是很常见的。那么这些是什么呢?这些参数用于通过方法传递变量、未知数量的参数。它们之间的区别在于 args 用于传递非关键字参数,其中 kwargs 是关键字参数(因此在名称中进行网格划分以使其成为 kwargs)。参数是典型参数。Kwargs,基本上是字典。你可以把kwargs看作是正在传递的词典。

所以,从理论上讲,你可以有一个类似于 def example(farg, *, **) 的方法或函数。Fargs 是必需的,正如您现在可能知道的那样,如果没有为它们分配任何内容,它们将抛出错误。

接下来,我们在 def __init___下有以下行:

        tk.Tk.__init__(self, *args, **kwargs)
		

在这里,我们初始化继承的类。现在来快速查看一些特定于 Tkinter 的代码:

        container = tk.Frame(self)
        container.pack(side="top", fill="both", expand=True)
        container.grid_rowconfigure(0, weight=1)
        container.grid_columnconfigure(0, weight=1)
		

        我们已经定义了这个容器,它将填充一堆帧,以便稍后访问。

        接下来,我们有 container.pack。在 Tkinter 中,有两种主要方法可以填充和定位您在框架中创建的小部件。一种方式是包装,另一种是网格。根据您的需求和您对什么感到满意,您可能会比另一个更多地使用一个。在大多数情况下,我发现网格让我对应用程序有了最大的控制权。网格允许您创建一种网格,用于定向事物在应用程序中的位置。打包允许一些控制,但对我来说,主要是感觉你只是把东西塞进枕套里,只是尽力选择一边,但它并不总是按预期工作。

        grid_configure选项只是我们早期设置的一些简单配置设置。

        self.frames = {}
        frame = StartPage(container, self)
        self.frames[StartPage] = frame
		

        我们预定义了一个字典,该字典目前是空的。还记得之前关于字典和 kwargs 的事情吗?你认为这本词典会去哪里?

        接下来,我们定义框架是什么。最终,我们将用一堆可能的帧打包 self.frames,“顶部”帧是当前帧。现在,我们只有一个页面,即“StartPage”(尚未定义)。

        接下来,仍然在__init__中,我们有:

        frame.grid(row=0, column=0, sticky="nsew")
		

        在这里,我们使用网格来放置我们的小部件。对于此小组件,行和列均为 0。然后我们有粘性。“新南威尔士州”对应于方向(北、南、东、西)。粘性的概念就像对齐一样,拉伸略有变化。因此,如果您将某些内容对齐 e,那么小部件将在右侧。如果你说 sticky=“ew”,那么小部件将从左侧延伸到右侧。如果您像我们一样粘住“nsew”,那么将鼓励小部件填充分配的整个空间。

        self.show_frame(StartPage)
		

        最后,我们调用 show_frame,这是一个我们尚未定义的方法,但它将用于带来我们选择的框架,因此让我们创建该方法:

    def show_frame(self, cont):
        frame = self.frames[cont]
        frame.tkraise()
		

        另一种方法,带有 self 和控制器的 cont 参数。

        然后,我们将 frame 定义为 self.frame(即上面的字典),然后是控制器,它是字典中值的关键,即我们的框架。

        最后,我们执行 frame.tkraise(),它会将我们的框架带到顶部供用户查看。

太好了,我们的后端基本上都准备好了。现在让我们制作起始页。

class StartPage(tk.Frame):

    def __init__(self, parent, controller):

        tk.Frame.__init__(self, parent)
        label = tk.Label(self, text="This is the start page", font=LARGE_FONT)
        label.pack(pady=10,padx=10)
		

        在这里,我们有继承自 tk 的 StartPage 类。框架。然后我们有一个典型的__init__,其初始化为 tk。框架也是如此。

        然后我们定义一个标签小部件,即 Tkinter 代码。你会看到,对于“字体”,我们称之为Large_Font。这是一个常量,我们将把它放在应用程序的顶部。

        因此,在调用要导入的 tkinter 后,添加:

        LARGE_FONT= ("Verdana", 12)
		

        现在,回到我们的 StartPage 类。

        我们使用的是 .pack,在 y 和 x 上有一些填充。填充只是在事物的边缘添加一些空白区域,以帮助事物看起来不那么杂乱。

        最后,在脚本的最后,我们只需要:

app = SeaofBTCapp()
app.mainloop()
		

        App 是 SeaofBTCapp 类的对象,然后我们运行 .mainloop(),这是一个 tkinter 功能,但由于继承,我们可以使用它。

        大功告成!继续运行脚本,您应该看到:

Python 中的 Tkinter GUI 教程

3.2 完整脚本 

import tkinter as tk

LARGE_FONT= ("Verdana", 12)

class SeaofBTCapp(tk.Tk):

    def __init__(self, *args, **kwargs):

        tk.Tk.__init__(self, *args, **kwargs)
        container = tk.Frame(self)

        container.pack(side="top", fill="both", expand = True)

        container.grid_rowconfigure(0, weight=1)
        container.grid_columnconfigure(0, weight=1)

        self.frames = {}

        frame = StartPage(container, self)

        self.frames[StartPage] = frame

        frame.grid(row=0, column=0, sticky="nsew")

        self.show_frame(StartPage)

    def show_frame(self, cont):

        frame = self.frames[cont]
        frame.tkraise()

class StartPage(tk.Frame):

    def __init__(self, parent, controller):
        tk.Frame.__init__(self,parent)
        label = tk.Label(self, text="Start Page", font=LARGE_FONT)
        label.pack(pady=10,padx=10)

app = SeaofBTCapp()
app.mainloop()

四、总结

        我承认,在这一点上,可能会觉得 Tkinter 将非常难用,我们本可以更轻松地创建我们刚刚创建的窗口。我想强调的是,我们在这里所做的主要是为扩张奠定一些基础。我们现在有一个后端,可以很容易地让我们添加越来越多的页面,因此,在未来,添加页面就像创建另一个类一样简单,就像 StartPage 一样,向它添加导航,然后你就设置好了。

参考资料:

开发完整的 Tkinter 面向对象应用程序 (pythontutorial.net)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/209382.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

三、Zookeeper数据模型

目录 1、Znode兼具文件和目录两种特点 2、Znode具有原子性操作 3、Znode存储数据大小有限制 4、Znode通过路径引用 如下图中的每个节点称为一个Znode, 每个Znode由3部分组成: ZooKeeper的数据模型,在结构上和标准文件系统的非常相似,拥有…

力扣15题 三数之和 双指针算法

15. 三数之和 给你一个整数数组 nums ,判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i ! j、i ! k 且 j ! k ,同时还满足 nums[i] nums[j] nums[k] 0 。请 你返回所有和为 0 且不重复的三元组。 注意:答案中不可以包含重复的三…

013 OpenCV copyMakeBorder(padding)

目录 一、环境 二、原理 三、完整代码 一、环境 本文使用环境为: Windows10Python 3.9.17opencv-python 4.8.0.74 二、原理 cv.copyMakeBorder是OpenCV库中的一个函数,用于在图像周围添加边框(padding)。这个函数可以用于图…

Mongodb 开启oplog,java监听oplog并写入关系型数据库

开启Oplog windows mongodb bin目录下找到配置文件/bin/mongod.cfg,配置如下: replication:replSetName: localoplogSizeMB: 1024双击mongo.exe 执行 rs.initiate({_id: "local", members: [{_id: 0, host: "localhost:27017"}]})若出现如…

人工智能|机器学习——感知器算法原理与python实现

感知器算法是一种可以直接得到线性判别函数的线性分类方法,它是基于样本线性可分的要求下使用的。 一、线性可分与线性不可分 为了方便讨论,我们蒋样本增加了以为常数,得到增广样向量 y(1;;;...;),则n个样本的集合为&a…

[论文精读]利用大语言模型对扩散模型进行自我修正

本博客是一篇最新论文的精读,论文为UC伯克利大学相关研究者新近(2023.11.27)在arxiv上上传的《Self-correcting LLM-controlled Diffusion Models》 。 内容提要: 现有的基于扩散的文本到图像生成模型在生成与复杂提示精确对齐的图像时仍然存在困难,尤其是需要数值和…

ansible模块

目录 一、ansible的command模块 1.ad-hoc 2.playbook 3.command模块 二、ansible的shell模块 1.shell模块帮助 2.shell模块支持的参数和解释 3.简单试验 4.批量远程执行脚本 三、script模块 1.script模块帮助 2.shell模块支持的参数和解释 3.实践 四、ansible文件…

【pytorch】从yolo的make_grid理解torch.meshgrid、torch.stack

文章目录 简述1、torch.meshgrid 创建行列坐标2、torch.stack 结合行列坐标3、通过view函数扩展维度 简述 yolo检测 make_grid创建网格代码如下,那么什么是torch.meshgrid? def _make_grid(nx20, ny20):yv, xv torch.meshgrid([torch.arange(ny), torch.arange(…

C++基础 -28- 友元

友元用于访问类中的所有数据成员 类中的私有成员,类外不可访问 定义友元的格式(友元函数必须要在类内,声明) friend void show(person &b); 使用友元访问类的所有成员 #include "iostream"using namespace std…

深入Spring Security魔幻山谷-获取认证机制核心原理讲解(新版)

文/朱季谦 这是一个古老的传说。 在神秘的Web系统世界里,有一座名为Spring Security的山谷,它高耸入云,蔓延千里,鸟飞不过,兽攀不了。这座山谷只有一条逼仄的道路可通。然而,若要通过这条道路前往另一头的…

html实现各种好看的鼠标滑过图片特效模板

文章目录 1.鼠标悬浮效果1.1 渐动效果1.2 渐变效果1.3 边框效果1.4 线行效果1.5 图标效果1.6 块状效果1.7 边线效果1.8 放大效果1.9 渐出效果1.10 痕迹效果1.11 交叉效果1.12 着重效果1.13 详展效果1.14 能动效果1.15 明细效果 2.主要源码2.1 源代码 源码下载 作者:…

基于python的FMCW雷达工作原理仿真

这篇文章将介绍如何使用python来实现FMCW工作原理的仿真,第1章内容将介绍距离检测原理,第2章内容会介绍速度检测原理。 第1章 第1部分: 距离检测原理 调制的连续波雷达通常也被叫做调频连续波(FMCW)雷达是一个使用频率调制来测量…

鸿蒙(HarmonyOS)应用开发——容器组件(Grid组件)

前言 前面一篇文章中,已经说了List组件。那么接下来就是容器组件中的Grid组件 #mermaid-svg-oz1b7w45ASmMlZFa {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-oz1b7w45ASmMlZFa .error-icon{fill:#5522…

Docker-compose的在线与离线安装方式及问题解决

文章目录 一、在线方式1、GitHub2、daocloud.io 二、离线方式(推荐)三、验证 一、在线方式 1、GitHub curl -L "https://github.com/docker/compose/releases/latest/download/docker-compose-$(uname -s)-$(uname -m)" -o /usr/local/bin/d…

springboot缓存技术-Ehcache-Redis-memcached

springboot缓存技术-Ehcache-Redis-memcached 文章目录 springboot缓存技术-Ehcache-Redis-memcachedspring缓存使用方式手机验证码案例缓存供应商变更Ehcache变更缓存供应商Redis缓存供应商变更memcached下载安装memcachedSpringBoot整合memcached spring缓存使用方式 导缓存…

linux后端基础---笔记整理(tmux、vim、shell、ssh/scp、git、thrift、docker)

目录 1.Linux常用文件管理命令 2.tmux终端复用器/vim命令式文本编辑器 3.Shell语法 3.1 Shell—版本3.2 新建一个test.sh文件3.3 Shell文件—运行方式3.4 Shell—注释3.5 Shell—变量3.6 Shell—默认变量,文件参数, “$”的用法3.7 Shell—数组3.8 shell—expr命令…

java餐饮刀削面快餐店点餐服务系统springboot+jsp

网上点餐省去了客户很多不必要的时间和麻烦,给商家带来更多利益。同时,网上点餐可以辅助餐饮企业营销。传统的点餐是需要配备一个专业的服务员负责菜品介绍并记录顾客点单,确认后上交至后台厨房,厨房根据菜品种类安排做菜顺序最终…

数据结构:带头双向循环链表的实现

引言 单链表存在缺陷:需要从头开始找前一个节点 解决方法:双向链表 链表的结构(8种): 1. 单向,双向 2. 带头、不带头 带头即为带哨兵位的头节点,第一个节点不存储有效数据。带头节点&#…

垃圾回收与内存泄漏

前端面试大全JavaScript垃圾回收与内存泄漏 🌟经典真题 🌟什么是内存泄露 🌟JavaScript 中的垃圾回收 🌟标记清除 🌟引用计数 🌟真题解答 🌟总结 🌟经典真题 请介绍一下 Jav…

混合使用Windows和Linux子系统的工具和命令

文章目录 在Windows中运行Linux命令使用PowerShell混合使用Linux和Windows命令通过power shell在Windows混合使用Linux工具在Linux中混合使用Windows 工具 推荐阅读 Windows和Linux的工具和命令可以通过WSL互换使用。 可以在Linux子系统中运行Windows命令,也可以在W…