我是這樣學(xué)習(xí)GUI的——GUI入門(mén)之用Python寫(xiě)一個(gè)GUI程序并生成exe

?接上一篇matlab寫(xiě)的GUI程序,由于移植性差,準(zhǔn)備用python實(shí)現(xiàn)。

我是這樣學(xué)習(xí)GUI的——GUI入門(mén)之用Matlab寫(xiě)一個(gè)GUI程序并生成exe

Python可視化包比較多,由于tkinter是python自帶的包,不需要額外安裝,所以選擇tkinter來(lái)實(shí)現(xiàn)。

同樣先將輸入輸出部件可視化排布好。主要部件有窗口(root),畫(huà)布(canvas),圖像(Figure),輸入框(Entry),標(biāo)簽(Label),按鈕(Button)。

效果如下:

各個(gè)函數(shù)和整體結(jié)構(gòu):

一共寫(xiě)了七個(gè)函數(shù),分別是:初始化,顯示輸入框,顯示計(jì)算結(jié)果,獲取輸入值,計(jì)算,清除,主函數(shù)。

整個(gè)邏輯很簡(jiǎn)單,下面簡(jiǎn)單介紹一下各部分作用。


首先導(dǎo)入所需包,主要是matplotlib和tkinter。Matplotlib用于畫(huà)圖,tkinter用于整個(gè)界面包括各個(gè)控件的顯示。

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

一、定義初始化函數(shù)

這部分由于顯示標(biāo)簽并賦予初始化值,如下圖。

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值顯示函數(shù)

這部分定義y值輸入框并且把各個(gè)控件對(duì)應(yīng)變量的值顯示出來(lái),避免顯示值與實(shí)際值不一致的問(wèn)題。

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)

三、定義顯示斜率函數(shù)

與上個(gè)函數(shù)類(lèi)似,這個(gè)函數(shù)用于顯示斜率。

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)

四、定義獲取數(shù)據(jù)函數(shù)

將用戶(hù)輸入的y值賦給對(duì)應(yīng)變量。

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

五、定義計(jì)算執(zhí)行函數(shù)

當(dāng)按下calculate按鈕時(shí),開(kāi)始執(zhí)行該函數(shù)。

首先調(diào)用get_num()獲取數(shù)據(jù),接著計(jì)算slope,然后顯示slope,最后畫(huà)圖顯示出來(lái)。

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()

六、定義清除執(zhí)行函數(shù)

調(diào)用初始化函數(shù),然后顯示y和slope初始值。

def clear():

ini_fun()

show_y()

show_slope()

七、定義主函數(shù)

這部分定義主窗口,畫(huà)布,圖形以及計(jì)算和清除按鈕。

窗口循環(huán)語(yǔ)句root.mainloop()一定不能丟,不然無(wú)法顯示窗口,它的作用維持窗口刷新等待用戶(hù)交互。

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()

細(xì)心的同學(xué)已經(jīng)發(fā)現(xiàn),我在函數(shù)內(nèi)部用了很多global關(guān)鍵字,它用來(lái)聲明全局變量,這樣在函數(shù)內(nèi)部對(duì)變量做的更改計(jì)算才能傳遞給其他函數(shù)使用。

當(dāng)然,本著優(yōu)雅的原則,我又把它打包成exe了。

打包方法,安裝pyinstaller后,切換目錄到py文件所在目錄,運(yùn)行

pyinstaller –F –w [file name].py

提示完成在dist文件夾下生成exe文件即可雙擊運(yùn)行。

效果就這樣:

程序圖標(biāo)可以自定義,感興趣的可以搜索一下試試,很簡(jiǎn)單。

由于第一次獨(dú)立寫(xiě)python的GUI程序,為了簡(jiǎn)單,使用的grid方式定位,所以整體的布置效果不是太好。函數(shù)劃分邏輯也不是太好,對(duì)于這個(gè)邏輯簡(jiǎn)單的程序沒(méi)必要分這么多函數(shù)。

后續(xù)會(huì)選擇使用place定位方式,這樣各部件都可以精確定位。部件的樣式也可以重新設(shè)計(jì)一下。

可以在weiyoumimi公眾號(hào)回復(fù)“tkinter slope”獲取exe文件

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,505評(píng)論 6 533
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,556評(píng)論 3 418
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 176,463評(píng)論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 63,009評(píng)論 1 312
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,778評(píng)論 6 410
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 55,218評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,281評(píng)論 3 441
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,436評(píng)論 0 288
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,969評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,795評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,993評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,537評(píng)論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,229評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,659評(píng)論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 35,917評(píng)論 1 286
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,687評(píng)論 3 392
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,990評(píng)論 2 374

推薦閱讀更多精彩內(nèi)容