Tkinter 面向对象框架《二》

一、说明

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

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

二、第一个实验演示

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

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

import tkinter as tk
from tkinter import ttk
from tkinter.messagebox import showerrorclass TemperatureConverter:@staticmethoddef 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 optionsoptions = {'padx': 5, 'pady': 5}# temperature labelself.temperature_label = ttk.Label(self, text='Fahrenheit')self.temperature_label.grid(column=0, row=0, sticky=tk.W, **options)# temperature entryself.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.convertself.convert_button.grid(column=2, row=0, sticky=tk.W, **options)# result labelself.result_label = ttk.Label(self)self.result_label.grid(row=1, columnspan=3, **options)# add padding to the frame and show itself.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 showerrorclass TemperatureConverter:@staticmethoddef fahrenheit_to_celsius(f):return (f - 32) * 5 / 9class ConverterFrame(ttk.Frame):def __init__(self, container):super().__init__(container)# field optionsoptions = {'padx': 5, 'pady': 5}# temperature labelself.temperature_label = ttk.Label(self, text='Fahrenheit')self.temperature_label.grid(column=0, row=0, sticky=tk.W, **options)# temperature entryself.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.convertself.convert_button.grid(column=2, row=0, sticky=tk.W, **options)# result labelself.result_label = ttk.Label(self)self.result_label.grid(row=1, columnspan=3, **options)# add padding to the frame and show itself.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 tkclass 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)

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

测评补单助力亚马逊,速卖通,国际站卖家抢占市场,提升转化和评分

想要快速提升商品的销量,测评补单这种方法见效是最快的。特别是新品上线,缺少用户评价,转化率不好,很多商家新品上线都会做测评补单,搞些商品好评,不但可以提升转化,同时在平台也可以获得更多展…

向光有光megauging使用说明书(二,轻量级的visionpro)

测试程序暂时支持80万(包含1024*768)以上的gige工业相机 我们程序中使用注意力机制,其实就是感兴趣区域(roi,你看过我前面博文,就应该明白)精神的延伸,我们只处理全景图中的1024*76…

抑郁症由什么引起?

抑郁症的发生并不是单一原因所导致,而是多种因素相互作用的结果。以下是一些主要的原因: 首先,生物学因素在抑郁症的发病中起到了关键作用。研究显示,抑郁症可能与遗传有关,家族中有患抑郁症的成员会增加个体患病的风…

Windows的常用cmd命令总结

文章目录 一.盘符切换二: cd命令(打开文件/文件夹)三:查看目录四.创建和删除文件夹五.查看本机ip地址六.清除当前屏幕七.复制文件到另一个地方八.移动文件到另一个地方九.删除文件(不能删除文件夹)十.测试网络连接十一.停止任务进程Windows快捷键总结大全…

Java研学-IO流(三)

六 字节流 – 字节输出流系列 OutPutStream体系 1 OutPutStream系列 – 字节输出流 // 表示字节输出流所有类的超类,输出流接受输出字节并将其发送到某个接收器 public abstract class OutputStreamFileOutputStream/BufferedOutputStream 2 FileOutputStream类设…

Elasticsearch分词器--空格分词器(whitespace analyzer)

介绍 文本分析,是将全文本转换为一系列单词的过程,也叫分词。analysis是通过analyzer(分词器)来实现的,可以使用Elasticearch内置的分词器,也可以自己去定制一些分词器。除了在数据写入时将词条进行转换,那么在查询的时…

2023-简单点-机器学习中矩阵向量求导

机器学习中矩阵向量求导的概念是什么? 在机器学习中,矩阵向量求导的概念主要涉及对函数中的矩阵或向量参数进行求导运算。这种求导运算可以帮助我们了解函数值随参数的变化情况,进而应用于优化算法中。具体来说,当损失函数是一个…

ElasticSearch知识体系详解

1.介绍 ElasticSearch是基于Lucene的开源搜索及分析引擎,使用Java语言开发的搜索引擎库类,并作为Apache许可条款下的开放源码发布,是当前流行的企业级搜索引擎。 它可以被下面这样准确的形容: 一个分布式的实时文档存储&#xf…

图解java.util.concurrent并发包源码系列——深入理解定时任务线程池ScheduledThreadPoolExecutor

深入理解定时任务线程池ScheduledThreadPoolExecutor ScheduledThreadPoolExecutor作用与用法ScheduledThreadPoolExecutor内部执行流程DelayedWorkQueueScheduledFutureTask源码分析任务提交ScheduledFutureTask的属性和方法delayedExecute(t) 任务执行ScheduledFutureTask.su…

软件测试计划书

测试计划书 1.测试参考文档和测试提交文档 2.测试进度计划 3.测试资源 4.系统风险、优先级 5.测试策略 6.缺陷管理 7.测试停止标准 软件开发全文档下载进入主页。

Elasticsearch 如何处理 Aggs 顺序中的大写字母和小写字母?

Elasticsearch 排序允许你根据特定条件对搜索结果进行排序。 然而,在排序时处理区分大小写时,Elasticsearch 将大写和小写字母视为不同的字符,分别对它们进行排序。 这是因为 ASCII 表顺序是从大写 A 到小写 z。 默认情况下,Elas…

微信小程序 地图撒点

1. 微信小程序 地图撒点 1.1 说明 首先使用微信小程序自带标签,并且设置好宽高让地图显示,用longitude和latitude表示中心点。   (1)show-location 显示带有方向的当前定位点,本项目不需要不添加。   (2&#xff…