我是这样学习GUI的——GUI入门之用Python写一个GUI程序并生成exe

 接上一篇matlab写的GUI程序,由于移植性差,准备用python实现。

我是这样学习GUI的——GUI入门之用Matlab写一个GUI程序并生成exe

Python可视化包比较多,由于tkinter是python自带的包,不需要额外安装,所以选择tkinter来实现。

同样先将输入输出部件可视化排布好。主要部件有窗口(root),画布(canvas),图像(Figure),输入框(Entry),标签(Label),按钮(Button)。

效果如下:

各个函数和整体结构:

一共写了七个函数,分别是:初始化,显示输入框,显示计算结果,获取输入值,计算,清除,主函数。

整个逻辑很简单,下面简单介绍一下各部分作用。


首先导入所需包,主要是matplotlib和tkinter。Matplotlib用于画图,tkinter用于整个界面包括各个控件的显示。

import matplotlib

matplotlib.use('TkAgg')

from matplotlib.backends.backend_tkagg import FigureCanvasTkAgg, NavigationToolbar2Tk

from matplotlib.figure import Figure

from tkinter import *

import tkinter as tk

import numpy as np

一、定义初始化函数

这部分由于显示标签并赋予初始化值,如下图。

Grid方法用于以表格的形式定位元素。

def ini_fun():

global slope1, slope2, slope3, slope4, slope5, slope6, slope7, slope8, slope9

global y0, y1, y2, y3, y4, y5, y6, y7, y8, y9, inter_width, inter_width_entry

y0_label = tk.Label(root, text='y0').grid(row=3, column=0)

y1_label = tk.Label(root, text='y1').grid(row=3, column=1)

y2_label = tk.Label(root, text='y2').grid(row=3, column=2)

y3_label = tk.Label(root, text='y3').grid(row=3, column=3)

y4_label = tk.Label(root, text='y4').grid(row=3, column=4)

y5_label = tk.Label(root, text='y5').grid(row=3, column=5)

y6_label = tk.Label(root, text='y6').grid(row=3, column=6)

y7_label = tk.Label(root, text='y7').grid(row=3, column=7)

y8_label = tk.Label(root, text='y8').grid(row=3, column=8)

y9_label = tk.Label(root, text='y9').grid(row=3, column=9)

slope1_label = tk.Label(root, text='slope1').grid(row=6, column=1)

slope2_label = tk.Label(root, text='slope2').grid(row=6, column=2)

slope3_label = tk.Label(root, text='slope3').grid(row=6, column=3)

slope4_label = tk.Label(root, text='slope4').grid(row=6, column=4)

slope5_label = tk.Label(root, text='slope5').grid(row=6, column=5)

slope6_label = tk.Label(root, text='slope6').grid(row=6, column=6)

slope7_label = tk.Label(root, text='slope7').grid(row=6, column=7)

slope8_label = tk.Label(root, text='slope8').grid(row=6, column=8)

slope9_label = tk.Label(root, text='slope9').grid(row=6, column=9)

inter_width_label = tk.Label(root, text='inter_width:').grid(row=4, column=2)

inter_width_entry = tk.Entry(root, width=4)

inter_width_entry.grid(row=4, column=3)

inter_width_entry.insert(1,'10')

slope1 = '0.0'

slope2 = '0.0'

slope3 = '0.0'

slope4 = '0.0'

slope5 = '0.0'

slope6 = '0.0'

slope7 = '0.0'

slope8 = '0.0'

slope9 = '0.0'

y0 = 0

y1 = 0

y2 = 0

y3 = 0

y4 = 0

y5 = 0

y6 = 0

y7 = 0

y8 = 0

y9 = 0

二、定义y值显示函数

这部分定义y值输入框并且把各个控件对应变量的值显示出来,避免显示值与实际值不一致的问题。

Entry控件的insert方法将变量值显示在控件中。

def show_y():

global y0_entry, y1_entry, y2_entry, y3_entry, y4_entry, y5_entry, y6_entry, y7_entry, y8_entry, y9_entry

global y0, y1, y2, y3, y4, y5, y6, y7, y8, y9, inter_width

y0_entry = tk.Entry(root, width=4)

y0_entry.grid(row=2, column=0)

y0_entry.insert(1,y0)

y1_entry = tk.Entry(root, width=4)

y1_entry.insert(1,y1)

y1_entry.grid(row=2, column=1)

y2_entry = tk.Entry(root, width=4)

y2_entry.grid(row=2, column=2)

y2_entry.insert(1,y2)

y3_entry = tk.Entry(root, width=4)

y3_entry.grid(row=2, column=3)

y3_entry.insert(1,y3)

y4_entry = tk.Entry(root, width=4)

y4_entry.grid(row=2, column=4)

y4_entry.insert(1,y4)

y5_entry = tk.Entry(root, width=4)

y5_entry.grid(row=2, column=5)

y5_entry.insert(1,y5)

y6_entry = tk.Entry(root, width=4)

y6_entry.grid(row=2, column=6)

y6_entry.insert(1,y6)

y7_entry = tk.Entry(root, width=4)

y7_entry.grid(row=2, column=7)

y7_entry.insert(1,y7)

y8_entry = tk.Entry(root, width=4)

y8_entry.grid(row=2, column=8)

y8_entry.insert(1,y8)

y9_entry = tk.Entry(root, width=4)

y9_entry.grid(row=2, column=9)

y9_entry.insert(1,y9)

三、定义显示斜率函数

与上个函数类似,这个函数用于显示斜率。

def show_slope():

slope1_num = tk.Label(root, text=slope1).grid(row=5, column=1)

slope2_num = tk.Label(root, text=slope2).grid(row=5, column=2)

slope3_num = tk.Label(root, text=slope3).grid(row=5, column=3)

slope4_num = tk.Label(root, text=slope4).grid(row=5, column=4)

slope5_num = tk.Label(root, text=slope5).grid(row=5, column=5)

slope6_num = tk.Label(root, text=slope6).grid(row=5, column=6)

slope7_num = tk.Label(root, text=slope7).grid(row=5, column=7)

slope8_num = tk.Label(root, text=slope8).grid(row=5, column=8)

slope9_num = tk.Label(root, text=slope9).grid(row=5, column=9)

四、定义获取数据函数

将用户输入的y值赋给对应变量。

def get_num():

global y0, y1, y2, y3, y4, y5, y6, y7, y8, y9, inter_width, inter_width_entry

y0 = int(y0_entry.get())

y1 = int(y1_entry.get())

y2 = int(y2_entry.get())

y3 = int(y3_entry.get())

y4 = int(y4_entry.get())

y5 = int(y5_entry.get())

y6 = int(y6_entry.get())

y7 = int(y7_entry.get())

y8 = int(y8_entry.get())

y9 = int(y9_entry.get())

inter_width = int(inter_width_entry.get())

return y0, y1, y2, y3, y4, y5, y6, y7, y8, y9, inter_width

五、定义计算执行函数

当按下calculate按钮时,开始执行该函数。

首先调用get_num()获取数据,接着计算slope,然后显示slope,最后画图显示出来。

def calculate():

global slope1, slope2, slope3, slope4, slope5, slope6, slope7, slope8, slope9

global y0, y1, y2, y3, y4, y5, y6, y7, y8, y9, inter_width

get_num()

slope1 = (y1 - y0) / inter_width

slope2 = (y2 - y1) / inter_width

slope3 = (y3 - y2) / inter_width

slope4 = (y4 - y3) / inter_width

slope5 = (y5 - y4) / inter_width

slope6 = (y6 - y5) / inter_width

slope7 = (y7 - y6) / inter_width

slope8 = (y8 - y7) / inter_width

slope9 = (y9 - y8) / inter_width

show_slope()

x = range(0, inter_width*10, inter_width)

y = [y0, y1, y2, y3, y4, y5, y6, y7, y8, y9]

f_plot.clear()

f_plot.plot(x, y)

canvs.draw()

六、定义清除执行函数

调用初始化函数,然后显示y和slope初始值。

def clear():

ini_fun()

show_y()

show_slope()

七、定义主函数

这部分定义主窗口,画布,图形以及计算和清除按钮。

窗口循环语句root.mainloop()一定不能丢,不然无法显示窗口,它的作用维持窗口刷新等待用户交互。

def main():

global root, f, f_plot, canvs

root = Tk()

root.geometry("800x700+10+10")

root.title("tkinter and matplotlib")

f = Figure(figsize=(7, 5), dpi=100)

f_plot = f.add_subplot(111)

canvs = FigureCanvasTkAgg(f, root)

canvs.get_tk_widget().grid(row=0, column=0, rowspan=1, columnspan=10, sticky="nesw")

Button(root, text='calculate', command=calculate).grid(row=4, column=4)

Button(root, text='clear', command=clear).grid(row=4, column=5)

ini_fun()

show_y()

show_slope()

root.mainloop()

if __name__ == '__main__':

main()

细心的同学已经发现,我在函数内部用了很多global关键字,它用来声明全局变量,这样在函数内部对变量做的更改计算才能传递给其他函数使用。

当然,本着优雅的原则,我又把它打包成exe了。

打包方法,安装pyinstaller后,切换目录到py文件所在目录,运行

pyinstaller –F –w [file name].py

提示完成在dist文件夹下生成exe文件即可双击运行。

效果就这样:

程序图标可以自定义,感兴趣的可以搜索一下试试,很简单。

由于第一次独立写python的GUI程序,为了简单,使用的grid方式定位,所以整体的布置效果不是太好。函数划分逻辑也不是太好,对于这个逻辑简单的程序没必要分这么多函数。

后续会选择使用place定位方式,这样各部件都可以精确定位。部件的样式也可以重新设计一下。

可以在weiyoumimi公众号回复“tkinter slope”获取exe文件

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,980评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,178评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,868评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,498评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,492评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,521评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,910评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,569评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,793评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,559评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,639评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,342评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,931评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,904评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,144评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,833评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,350评论 2 342

推荐阅读更多精彩内容