這將使你見過最全面的Python製作GUI學生管理系統教程

来源:https://www.cnblogs.com/hahaa/archive/2022/06/24/16408573.html
-Advertisement-
Play Games

表弟大學快畢業了,學了一個學期Python居然還不會寫學生管理系統,真的給我丟臉啊,教他又不肯學,還讓我直接給他寫,我真想兩巴掌上去,最終還是寫了給他,誰讓他是我表弟呢,關鍵時候還是得幫他一把! 寫完了放在那也是放著,所以今天分享給大家吧! 話不多說,咱們直接開始吧! 代碼解析 一、登錄頁面 1、定 ...


表弟大學快畢業了,學了一個學期Python居然還不會寫學生管理系統,真的給我丟臉啊,教他又不肯學,還讓我直接給他寫,我真想兩巴掌上去,最終還是寫了給他,誰讓他是我表弟呢,關鍵時候還是得幫他一把!

 

 

 


寫完了放在那也是放著,所以今天分享給大家吧!

話不多說,咱們直接開始吧!

 

 

代碼解析

一、登錄頁面

1、定義登錄類及初始化對象

首先導入咱們需要用到的模塊

from main import MainPage
# 文章看不懂,我專門錄了對應的視頻講解,完整代碼和視頻教程加群獲取。
# Python學習交流1裙:815624229  一群已滿加2群 ###
# Python學習交流2裙:279199867 ### 
 

 

登錄頁面

將畫板綁定到實例對象

self.root = master

 

self.page 畫紙 在屏幕上顯示一個矩形區域,多用來作為容器。

self.page = tk.Frame(self.root)
self.page.pack()
self.root.geometry("300x180")

 

tkinter 提供的可變變數,定義用戶名和密碼.

self.username = tk.StringVar()
self.password = tk.StringVar()

 

創建一個label

網格佈局

tk.Label(self.page).grid(row=0, column=0)
# textvariable 這個參數是把 tkinter 裡面的字元串變數與 空間綁定起來
tk.Label(self.page, text="賬戶").grid(row=1, column=0, stick=tk.E, pady=10)
tk.Entry(self.page, textvariable=self.username).grid(row=1, column=1, stick=tk.W, pady=10)
tk.Label(self.page, text="密碼").grid(row=2, column=0, stick=tk.E, pady=10)
tk.Entry(self.page, textvariable=self.password).grid(row=2, column=1, stick=tk.W, pady=10)

 

command 接受一個函數 執行登錄的邏輯

tk.Button(self.page, text="登錄", command=self.login_check).grid(row=3, column=0, stick=tk.W, pady=10)
tk.Button(self.page, text="退出", command=root.quit).grid(row=3, column=1, stick=tk.E, pady=10)

 

2、登錄函數

檢驗登錄

拿到賬號密碼

name = self.username.get()
pwd = self.password.get()

 

不去查詢資料庫

print(name, pwd)
if name == 'admin' and pwd == '123456':
    tkinter.messagebox.showinfo(title='恭喜',
                                message='登錄成功!')

 

摧毀當前頁面繪製的內容

self.page.destroy()

 

摧毀整個頁面繪製的內容

self.root.destroy()

 

頁面的切換

    MainPage(self.root)
else:
    tkinter.messagebox.showinfo(title='錯誤', message='賬戶或者密碼錯誤')

 

3、視窗調用

調用本文件方法,在本文件運行兩個輸入,在外面調用執行這個方法前面的數據。

創建一個對象,視窗對象,顯示界面.

if __name__ == '__main__':
    root = tk.Tk()
    LoginPage(root)
    root.mainloop()

 

二、主頁面顯示

1、定義頁面類,方便調用.

登錄界面

def __init__(self, master):
    self.root = master
    
    self.page = tk.Frame(self.root)
    self.page.pack()
    self.root.geometry("%dx%d" % (600, 400))
    self.create_page()

 

創建一個頂級菜單,顯示菜單.

def create_page(self):

    menubar = tk.Menu(self.root)

    menubar.add_command(label="錄入")
    menubar.add_command(label="查詢")
    menubar.add_command(label="刪除")
    menubar.add_command(label="修改")
    menubar.add_command(label="關於")

 

三 、頁面顯示

1、綁定各個頁面

寫在一起代碼太多,數據多,容易寫錯,混亂,可以寫一個文件專門來裝數據view.py

在view.py文件定義各個模塊的類

錄入

class InputFrame(tk.Frame):  # 繼承Frame類 
    def __init__(self, master):
        # 重新父類
        super().__init__(master)
        pass

 

查詢

class QueryFrame(tk.Frame):  # 繼承Frame類
    def __init__(self, master=None):
        super().__init__(master)
        pass

 

刪除

class DeleteFrame(tk.Frame):  # 繼承Frame類
    def __init__(self, master=None):
        super().__init__(master)

 

修改

class ChangeFrame(tk.Frame):  # 繼承Frame類
    def __init__(self, master=None):
        super().__init__(master)

 

關於

class AboutFrame(tk.Frame):  # 繼承Frame類
    def __init__(self, master=None):
        tk.Frame.__init__(self, master)
        self.root = master 

 

然後在main.py文件中綁定這些數據

self.input_page = InputFrame(self.root)
self.change_page = ChangeFrame(self.root)
self.query_page = QueryFrame(self.root)
self.delete_page = DeleteFrame(self.root)
self.about_page = AboutFrame(self.root)

 

2、錄入

在view.py文件中InputFrame類中添加相關內容,名字以及成績。

  • x = IntVar():整型變數,預設是0
  • x = DoubleVar():浮點型變數,預設是0.0
  • x = StringVar():字元串變數,預設是""
  • x = BooleanVar():布爾型變數,True是1,False是0
self.root = master  # 定義內部變數root
self.name = tk.StringVar()
self.math = tk.StringVar()
self.chinese = tk.StringVar()
self.english = tk.StringVar()
# 錄入
self.status = tk.StringVar()
# 調用create_page()函數
self.create_page()

 

編寫create_page()函數

def create_page(self):
    # pass
    # stick 控制項對象方向 tk.W 西方位
    # pady padding y 上下的寬度
    # row 行 表格佈局
    tk.Label(self).grid(row=0, stick=tk.W, pady=10)
    tk.Label(self, text='姓 名: ').grid(row=1, stick=tk.W, pady=10)
    # text variable 綁定控制項裡面的數據內容
    tk.Entry(self, textvariable=self.name).grid(row=1, column=1, stick=tk.E)
    tk.Label(self, text='數 學: ').grid(row=2, stick=tk.W, pady=10)
    tk.Entry(self, textvariable=self.math).grid(row=2, column=1, stick=tk.E)
    tk.Label(self, text='語 文: ').grid(row=3, stick=tk.W, pady=10)
    tk.Entry(self, textvariable=self.chinese).grid(row=3, column=1, stick=tk.E)
    tk.Label(self, text='英 語: ').grid(row=4, stick=tk.W, pady=10)
    tk.Entry(self, textvariable=self.english).grid(row=4, column=1, stick=tk.E)
    tk.Button(self, text='錄入', command=self.recode_student).grid(row=5, column=1, stick=tk.E, pady=10)
    tk.Label(self, textvariable=self.status).grid(row=6, column=1, stick=tk.E, pady=10)

 

在main.py中綁定這些數據

menubar.add_command(label="錄入", command=self.show_input)

def show_input(self):
    self.input_page.pack()
    # pack_forget()隱藏佈局
    # self.change_page.pack_forget()
    # self.query_page.pack_forget()
    # self.delete_page.pack_forget()
    # self.about_page.pack_forget()

 

在view.py文件中數據錄入

def recode_student(self):
    stu = {'name': self.name.get(), 'chinese': self.chinese.get(),
             'math': self.math.get(), 'english': self.english.get()}
    # 點擊錄入之後需要刷新頁面
    self.name.set('')
    self.chinese.set('')
    self.math.set('')
    self.english.set('')
    db.insert(stu)
    self.status.set('提交數據成功')
    print(stu)

 

插入數據及保存數據,可以寫一個文件保存,寫在一起代碼量太大,出現錯誤機率也大寫一個db.py文件.

class StudentsDB:
    def __init__(self):
        self.students = []

    def insert(self, student):
        self.students.append(student)


db = StudentsDB()

if __name__ == '__main__':
    print(db.students)

 

四、查詢數據

在QueryFrame()類添加數據

  • ttk.Treeview 樹視圖,百度一下

  • shows:
    headings
    tree
    data_list

  • columns
    值是一個列表。列表裡每個元素代表一個列標識符的名稱。列表的長度為列的長度。

繼承Frame類

class QueryFrame(tk.Frame):  
    def __init__(self, master=None):
        super().__init__(master)

 

定義內部變數root

self.root = master  #
columns = ('name', 'chinese', 'math', 'english')
self.tree_view = ttk.Treeview(self, show='headings', columns=columns)

 

每個數據大小每一格

self.tree_view.column('name', width=80, anchor='center')
self.tree_view.column('chinese', width=80, anchor='center')
self.tree_view.column('math', width=80, anchor='center')
self.tree_view.column('english', width=80, anchor='center')

 

上面的標簽及headings

self.tree_view.heading('name', text='姓名')
self.tree_view.heading('chinese', text='語文')
self.tree_view.heading('math', text='數學')
self.tree_view.heading('english', text='英語')
self.tree_view.pack(fill=tk.BOTH, expand=True)
tk.Button(self, text='刷新數據', command=self.show_data_frame).pack(anchor=tk.E, pady=5)
self.show_data_frame()

 

刷新數據,顯示數據.

def show_data_frame(self):
    # 刪除舊的階段
    for _ in map(self.tree_view.delete, self.tree_view.get_children('')):
        pass
    # 先要顯示所有數據 在db文件加入顯示數據代碼
    students = db.all()
    index = 0
    for stu in students:
        # print(stu)
        self.tree_view.insert('', index + 1, values=(
            stu['name'], stu['chinese'], stu['math'], stu['english'],
        ))

 

顯示數據

在db.py中加

def all(self):
    return self.students

 

view.py

刷新插入數據

  • 第一個參數:parent : 對於有樹欄的Treeview,parent是父節點,對於只是列表欄的Treeview,parent一般為空。
  • 第二個參數:index :插入位置。可以是END或’end’ ,也可以是數字的,如果你想新插入的item(記錄)成為第某節點的第一個,index就設為0,以此類推。
  • values:顯示的值及插入的數據,這一列數據。
self.tree_view.insert('', index + 1, values=(
                stu['name'], stu['chinese'], stu['math'],                                 stu['english'],
            ))

 

插入數據刷新後更新頁面

  • map(func, lst),將傳入的函數變數func作用到lst變數的每個元素中,並將結果組成新的列表(Python2)/迭代器(Python3)返回。
  • get_children(item=None)

返回一個item的所有子item,這個子item是一個列表形式,如果item沒指定,則返回根目錄的item

for _ in map(self.tree_view.delete, self.tree_view.get_children('')):
            pass

 

在main.py文件綁定數據

    menubar.add_command(label="查詢", command=self.show_all)

def show_all(self):
    # 隱藏佈局
    self.input_page.pack_forget()
    # self.change_page.pack_forget()
    self.query_page.pack()
    # self.delete_page.pack_forget()
    # self.about_page.pack_forget()

 

五、刪除數據

在DeleteFrame()類添加數據

class DeleteFrame(tk.Frame):  # 繼承Frame類
    def __init__(self, master=None):
        super().__init__(master)
        self.root = master  # 定義內部變數root
        tk.Label(self, text='刪除數據').pack()
        self.delete_frame = tk.Frame(self)
        self.delete_frame.pack()
        self.status = tk.StringVar()
        self.username = tk.StringVar()
        tk.Label(self.delete_frame, text='根據名字刪除信息').pack(anchor=tk.W, padx=20)
        tk.Entry(self.delete_frame, textvariable=self.username).pack(side=tk.LEFT, padx=20, pady=5)

        tk.Button(self.delete_frame, text='刪除', command=self._delete).pack()
        tk.Label(self, textvariable=self.status).pack()

 

1、點擊刪除,刪除數據

def _delete(self):
    username = self.username.get()
    flag, message = db.delete_by_name(username)
    self.status.set(message)

 

在db.py文件中添加刪除邏輯

def delete_by_name(self, name):
    for student in self.students:
        if name == student['name']:
            self.students.remove(student)
            return True, f'{name} 刪除成功'
    return False, f'{name} 不存在'

 

在main.py中綁定數據

    menubar.add_command(label="刪除", command=self.show_delete)
def show_delete(self):
    self.input_page.pack_forget()
    self.query_page.pack_forget()
    self.delete_page.pack()

 

六、修改數據

在ChangeFrame()類添加數據

self.root = master  # 定義內部變數root

        tk.Label(self, text='修改界面').pack()
        self.change_frame = tk.Frame(self)
        self.change_frame.pack()
        self.status = tk.StringVar()
        self.name = tk.StringVar()
        self.math = tk.StringVar()
        self.chinese = tk.StringVar()
        self.english = tk.StringVar()

        tk.Label(self.change_frame).grid(row=0, stick=tk.W, pady=1)
        tk.Label(self.change_frame, text='姓 名: ').grid(row=1, stick=tk.W, pady=10)
        tk.Entry(self.change_frame, textvariable=self.name).grid(row=1, column=1, stick=tk.E)
        tk.Label(self.change_frame, text='數 學: ').grid(row=2, stick=tk.W, pady=10)
        tk.Entry(self.change_frame, textvariable=self.math).grid(row=2, column=1, stick=tk.E)
        tk.Label(self.change_frame, text='語 文: ').grid(row=3, stick=tk.W, pady=10)
        tk.Entry(self.change_frame, textvariable=self.chinese).grid(row=3, column=1, stick=tk.E)
        tk.Label(self.change_frame, text='英 語: ').grid(row=4, stick=tk.W, pady=10)
        tk.Entry(self.change_frame, textvariable=self.english).grid(row=4, column=1, stick=tk.E)
        tk.Button(self.change_frame, text='查詢', command=self._search).grid(row=6, column=0, stick=tk.W, pady=10)
        tk.Button(self.change_frame, text='修改', command=self._change).grid(row=6, column=1, stick=tk.E, pady=10)
        tk.Label(self.change_frame, textvariable=self.status).grid(row=7, column=1, stick=tk.E, pady=10)

 

1、查詢數據

db.py文件中查詢數據邏輯

def search_by_name(self, name):
    for student in self.students:
        if name == student['name']:
            return True, student
    return False, f'{name} 不存在'

 

view.py文件點擊查詢,顯示數據

def _search(self):
    flag, info = db.search_by_name(self.name.get())
    if flag:
        self.name.set(info['name'])
        self.chinese.set(info['chinese'])
        self.math.set(info['math'])
        self.english.set(info['english'])
        self.status.set('數據查詢成功')
    else:
        # 直接返回錯誤的信息
        self.status.set(info)

 

2、修改數據及更新

db.py文件更新數據邏輯

def update(self, stu):
    name = stu['name']
    for student in self.students:
        if name == student['name']:
            student.update(stu)
            return True, f'{stu["name"]} 用戶數據修改成功'
    else:
        return False, f'{name} 不存在'

 

view.py文件修改數據

def _change(self):
    stu = {'name': self.name.get(), 'chinese': self.chinese.get(),
           'math': self.math.get(), 'english': self.english.get(), }
    self.name.set('')
    self.chinese.set('')
    self.math.set('')
    self.english.set('')
    db.update(stu)
    self.status.set('修改數據成功')

 

main.py文件進行綁定

    menubar.add_command(label="修改", command=self.show_change)
def show_change(self):
    self.input_page.pack_forget()
    self.query_page.pack_forget()
    self.delete_page.pack_forget()
    self.change_page.pack()

 

七、關於部分

view.py中在AboutFrame()添加數據

class AboutFrame(tk.Frame):  # 繼承Frame類
    def __init__(self, master=None):
        tk.Frame.__init__(self, master)
        self.root = master  # 定義內部變數root
        tk.Label(self, text='關於作品:本作品由 嗨學編程 製作').pack(anchor=tk.W)
        tk.Label(self, text='關於作者:嗨學編程').pack(anchor=tk.W)
        tk.Label(self, text='版權所有:嗨學編程').pack(anchor=tk.W)

 

main.py綁定數據

    menubar.add_command(label="關於", command=self.show_about)
def show_about(self):
    self.input_page.pack_forget()
    self.query_page.pack_forget()
    self.delete_page.pack_forget()
    self.change_page.pack_forget()
    self.about_page.pack()

 

八、數據保存,保存信息

db.py對數據保存

創建空json文件

import os

file = "students.json"
# 判斷文件是否存在,不存在則創建
if not os.path.exists(file):
    open(file, 'w')
    # 報錯
    # os.mknod(file)

 

  • Windows上的Python不支持mknod函數,因為在Windows上沒有node這個概念。

1、保存數據

def save_data(self):
    with open('students.json', mode='w', encoding='utf-8') as f:
        text = json.dumps(self.students, indent=2, ensure_ascii=False)
        f.write(text)

 

2 、讀取數據

def _load_students_data(self):
    with open('students.json', mode='r', encoding='utf-8') as f:
        text = f.read()
    if text:
        self.students = json.loads(text)

 

在view.py文件中調用保存函數,對完成後的數據進行保存。

錄入

def recode_student(self):
    db.save_data()

 

刪除

def _delete(self):
    db.save_data()

 

修改

def _change(self):
    db.save_data()

 

好了今天的分享就到這裡,兄弟們下次見!

大家覺得有幫助的話,順手點個關註+贊+收藏吧,愛你們!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、實驗目的 在信息時代高速發展的現在,“互聯網+”的使用日趨zhanzhang過互聯網學習知識,傳遞思想,溝通交流,在眾多數據和用戶的碰 撞中,互聯網經濟應運而生。學會利用網路收集信息是最基本的要求,接下來,我將以“行業網站”——站長之 家為例,通過Python爬取各個網站的信息(主要為名稱、Al ...
  • Go 語言入門練手項目系列 01 基於命令行的圖書的增刪查改 02 文件管理 持續更新中... > 本文來自博客園,作者:Arway,轉載請註明原文鏈接:https://www.cnblogs.com/cenjw/p/gobeginner-proj-bookstore-cli.html 介紹 這是一 ...
  • 1.路徑處理 1.找模塊:sys.path import sys print(sys.path) - 1.理解 - 1.是python去查找包或模塊 - 2.項目開始根目錄,python內置的目錄 - 3.雖然說python的安裝目錄下也可以存放我們寫的模塊,但是不建議(太多了,不大好找) - 4. ...
  • 本篇內容將在上一篇已有的內容基礎上,進一步的聊一下項目中使用JPA的一些高階複雜場景的實踐指導,覆蓋了主要核心的JPA使用場景,可以讓你在需求開發的時候對JPA的使用更加的游刃有餘。 ...
  • 前言 Steam是由美國電子游戲商Valve於2003年9月12日推出的數字發行平臺,被認為是電腦游戲界最大的數位發行平臺之一,Steam平臺是全球最大的綜合性數字發行平臺之一。玩家可以在該平臺購買、下載、討論、上傳和分享游戲和軟體。 而每周的steam會開啟了一輪特惠,可以讓游戲打折,而玩家就會 ...
  • Hi,大家好,我是Mic 一個工作5年的粉絲找到我。 他說: “Mic老師,你要是能回答出這個問題,我就佩服你” 我當場就懵了,現在打賭都這麼隨意了嗎? 我問他問題是什麼,他說“Kafka如何避免重覆消費的問題!” 下麵看看普通人和高手的回答! 普通人: Kafka怎麼避免重覆消費就是我們可以通過 ...
  • 前言 今天給大家分享一下我自己寫的筆記,純純的都是乾貨,關於字好像也能看。這是我學python整理出來的一些資料,希望對大家 有用。想要更多的資料那就的給一個關註了… python學習交流Q群:903971231### #導入Counter from collections import Count ...
  • Zookeeper3.7源碼剖析 能力目標 掌握Zookeeper中Session的管理機制 能基於Client進行Debug測試Session創建/刷新操作 能搭建Zookeeper集群源碼配置 掌握集群環境下Leader選舉啟動過程 能說出Zookeeper選舉過程中的概念 能說出Zookeep ...
一周排行
    -Advertisement-
    Play Games
  • Timer是什麼 Timer 是一種用於創建定期粒度行為的機制。 與標準的 .NET System.Threading.Timer 類相似,Orleans 的 Timer 允許在一段時間後執行特定的操作,或者在特定的時間間隔內重覆執行操作。 它在分散式系統中具有重要作用,特別是在處理需要周期性執行的 ...
  • 前言 相信很多做WPF開發的小伙伴都遇到過表格類的需求,雖然現有的Grid控制項也能實現,但是使用起來的體驗感並不好,比如要實現一個Excel中的表格效果,估計你能想到的第一個方法就是套Border控制項,用這種方法你需要控制每個Border的邊框,並且在一堆Bordr中找到Grid.Row,Grid. ...
  • .NET C#程式啟動閃退,目錄導致的問題 這是第2次踩這個坑了,很小的編程細節,容易忽略,所以寫個博客,分享給大家。 1.第一次坑:是windows 系統把程式運行成服務,找不到配置文件,原因是以服務運行它的工作目錄是在C:\Windows\System32 2.本次坑:WPF桌面程式通過註冊表設 ...
  • 在分散式系統中,數據的持久化是至關重要的一環。 Orleans 7 引入了強大的持久化功能,使得在分散式環境下管理數據變得更加輕鬆和可靠。 本文將介紹什麼是 Orleans 7 的持久化,如何設置它以及相應的代碼示例。 什麼是 Orleans 7 的持久化? Orleans 7 的持久化是指將 Or ...
  • 前言 .NET Feature Management 是一個用於管理應用程式功能的庫,它可以幫助開發人員在應用程式中輕鬆地添加、移除和管理功能。使用 Feature Management,開發人員可以根據不同用戶、環境或其他條件來動態地控制應用程式中的功能。這使得開發人員可以更靈活地管理應用程式的功 ...
  • 在 WPF 應用程式中,拖放操作是實現用戶交互的重要組成部分。通過拖放操作,用戶可以輕鬆地將數據從一個位置移動到另一個位置,或者將控制項從一個容器移動到另一個容器。然而,WPF 中預設的拖放操作可能並不是那麼好用。為瞭解決這個問題,我們可以自定義一個 Panel 來實現更簡單的拖拽操作。 自定義 Pa ...
  • 在實際使用中,由於涉及到不同編程語言之間互相調用,導致C++ 中的OpenCV與C#中的OpenCvSharp 圖像數據在不同編程語言之間難以有效傳遞。在本文中我們將結合OpenCvSharp源碼實現原理,探究兩種數據之間的通信方式。 ...
  • 一、前言 這是一篇搭建許可權管理系統的系列文章。 隨著網路的發展,信息安全對應任何企業來說都越發的重要,而本系列文章將和大家一起一步一步搭建一個全新的許可權管理系統。 說明:由於搭建一個全新的項目過於繁瑣,所有作者將挑選核心代碼和核心思路進行分享。 二、技術選擇 三、開始設計 1、自主搭建vue前端和. ...
  • Csharper中的表達式樹 這節課來瞭解一下表示式樹是什麼? 在C#中,表達式樹是一種數據結構,它可以表示一些代碼塊,如Lambda表達式或查詢表達式。表達式樹使你能夠查看和操作數據,就像你可以查看和操作代碼一樣。它們通常用於創建動態查詢和解析表達式。 一、認識表達式樹 為什麼要這樣說?它和委托有 ...
  • 在使用Django等框架來操作MySQL時,實際上底層還是通過Python來操作的,首先需要安裝一個驅動程式,在Python3中,驅動程式有多種選擇,比如有pymysql以及mysqlclient等。使用pip命令安裝mysqlclient失敗應如何解決? 安裝的python版本說明 機器同時安裝了 ...