那天我刚想下班摸鱼,办公室门一开,班主任冲进来一句:“东哥,你会不会搞那个啥……Python?帮我弄个成绩分析,我这Excel都要被我点炸了。” 我看了一眼她那张一万多行的表,脑袋“嗡”一下:这不就是典型背锅现场嘛。
我就边给她倒水边想:要不整一个小桌面程序,点点按钮就能算平均分、挂科名单、年级前十那种。对老师来说,浏览器都嫌复杂,还是老老实实用 Tkinter 弹个窗口,数据用 SQLite3 丢本地一个文件,拷U盘就能走人,谁用谁知道。
说干就干,我先随手写了个最关键的——建库的那个小函数。你们别嫌丑啊,我当时也是边写边改的:
import sqlite3
DB_PATH = "score.db"
definit_db():
conn = sqlite3.connect(DB_PATH)
cur = conn.cursor()
cur.execute("""
CREATE TABLE IF NOT EXISTS students (
id INTEGER PRIMARY KEY AUTOINCREMENT,
name TEXT NOT NULL,
math REAL NOT NULL,
chinese REAL NOT NULL,
english REAL NOT NULL
)
""")
conn.commit()
conn.close()
这个东西的好处就是——老师就算手一抖把 score.db 删了,下次点开程序,它自己又给你建一张新表,顶多数据没了,程序还在,属于“人没了系统还跑着”的那种顽强。
然后是最日常的需求:录入一个学生成绩。老师说她只想“输名字输分,点下按钮”,其他就别跟她谈什么CRUD这种吓人的词了。那我就照着她的口味来一个:
defadd_student(name, math, chinese, english):
conn = sqlite3.connect(DB_PATH)
cur = conn.cursor()
cur.execute(
"INSERT INTO students(name, math, chinese, english) VALUES (?, ?, ?, ?)",
(name, math, chinese, english)
)
conn.commit()
conn.close()
别看这几行,后来救了我一次命。老师有天激动地跟我说:“我刚刚不小心点了两次添加怎么办?” 我心里想完了完了,要不要给她做个撤销功能。冷静一下翻了下库,好家伙,她第二次输入的时候把数学从 95 改成了 96,还挺会“人工纠错”的,我直接装作“系统有智能纠偏能力”,这叫算发(算法)辅助教学,懂?
不过光能存还不行,老师最关心的是——谁考砸了,谁要叫家长。这个功能就有点阴间了,我一边写一边庆幸我已经毕业多年。先搞个基础统计,把平均分、最高分之类的算出来:
defcalc_stats():
conn = sqlite3.connect(DB_PATH)
cur = conn.cursor()
cur.execute("""
SELECT
COUNT(*) as total,
AVG(math), AVG(chinese), AVG(english),
MAX(math), MAX(chinese), MAX(english)
FROM students
""")
row = cur.fetchone()
conn.close()
return {
"total": row[0],
"avg_math": row[1],
"avg_chinese": row[2],
"avg_english": row[3],
"max_math": row[4],
"max_chinese": row[5],
"max_english": row[6],
}
写完我自己先跑了一遍,看到 AVG 出来的小数点后那么多位,脑子里就浮现出老师拿计算器对着小数位一位一位按的画面,于是后面 UI 那块我干脆给她都四舍五入好了,不给她留下“质疑程序”的空间。
说了半天数据库,得拉回到 Tkinter 这边。老师的要求只有一句话:“要能点,能看,别让我记命令。” 那没办法,得给她做个简单粗暴的窗体。那会儿我大概是这么糊出来的:
import tkinter as tk
from tkinter import messagebox
defbuild_ui():
root = tk.Tk()
root.title("学生成绩分析系统")
tk.Label(root, text="姓名").grid(row=0, column=0)
tk.Label(root, text="数学").grid(row=1, column=0)
tk.Label(root, text="语文").grid(row=2, column=0)
tk.Label(root, text="英语").grid(row=3, column=0)
name_var = tk.StringVar()
math_var = tk.StringVar()
chi_var = tk.StringVar()
eng_var = tk.StringVar()
tk.Entry(root, textvariable=name_var).grid(row=0, column=1)
tk.Entry(root, textvariable=math_var).grid(row=1, column=1)
tk.Entry(root, textvariable=chi_var).grid(row=2, column=1)
tk.Entry(root, textvariable=eng_var).grid(row=3, column=1)
result_box = tk.Text(root, width=40, height=10)
result_box.grid(row=0, column=2, rowspan=6, padx=10)
defon_add():
name = name_var.get().strip()
try:
math = float(math_var.get())
chi = float(chi_var.get())
eng = float(eng_var.get())
except ValueError:
messagebox.showerror("错误", "分数必须是数字,别夹中文逗号了…")
return
ifnot name:
messagebox.showwarning("提示", "姓名不能为空")
return
add_student(name, math, chi, eng)
messagebox.showinfo("成功", "已保存")
name_var.set("")
math_var.set("")
chi_var.set("")
eng_var.set("")
defon_stats():
data = calc_stats()
result_box.delete("1.0", tk.END)
if data["total"] == 0:
result_box.insert(tk.END, "还没有任何学生数据\n")
return
result_box.insert(tk.END, f"总人数:{data['total']}\n")
result_box.insert(tk.END, f"数学 平均:{data['avg_math']:.1f} 最高:{data['max_math']}\n")
result_box.insert(tk.END, f"语文 平均:{data['avg_chinese']:.1f} 最高:{data['max_chinese']}\n")
result_box.insert(tk.END, f"英语 平均:{data['avg_english']:.1f} 最高:{data['max_english']}\n")
tk.Button(root, text="添加成绩", command=on_add).grid(row=4, column=1, sticky="ew", pady=5)
tk.Button(root, text="统计分析", command=on_stats).grid(row=5, column=1, sticky="ew", pady=5)
return root
中间那个 result_box 文本框特别重要,一开始我偷懒用 messagebox 一直弹,老师用了一会儿直接跟我说:“你这个东西弹得我想砸电脑。” 我就赶紧改成右边一个大文本框,所有统计信息往里面刷,视觉上瞬间就温柔了很多。
你以为这样就结束了?并没有。真正的大坑在“挂科名单”那里。老师淡淡补了一句:“最好能看看谁挂了两门以上。” 好嘛,这一听就知道要写 SQL 条件,我边碎碎念边改代码,最后在原来的基础上加了个函数,算是比较暴力那种:
defget_failed_students(threshold=60):
conn = sqlite3.connect(DB_PATH)
cur = conn.cursor()
cur.execute("""
SELECT name, math, chinese, english
FROM students
""")
res = []
for name, m, c, e in cur.fetchall():
fail_count = sum(score < threshold for score in (m, c, e))
if fail_count >= 2:
res.append((name, m, c, e, fail_count))
conn.close()
return res
然后在 UI 里我又偷偷塞了个按钮:
defon_failed():
rows = get_failed_students()
result_box.delete("1.0", tk.END)
ifnot rows:
result_box.insert(tk.END, "没有两门以上不及格的学生,老师今天可以开心回家了。\n")
return
for name, m, c, e, cnt in rows:
line = f"{name} 数学:{m} 语文:{c} 英语:{e} 挂科数:{cnt}\n"
result_box.insert(tk.END, line)
tk.Button(root, text="挂科名单", command=on_failed).grid(row=6, column=1, sticky="ew", pady=5)
那天我把新版本拷到老师电脑上,她点开“挂科名单”,沉默了大概三秒钟,回头跟我说:“这个按钮你别给学生看到。” 我:懂,这叫内部运维工具,不对外开放。
最后就是把这些东西串一下,不然你们老是复制代码少一块又来问我“为什么没反应”。主函数那块就很朴素:
if __name__ == "__main__":
init_db()
app = build_ui()
app.mainloop()
整套下来也没多高大上,就是一个 Tkinter+SQLite3 的本地小系统,但胜在足够接地气: 老师双击一个图标,就能录入成绩、看平均分、看挂科名单;数据库只是一个 score.db 文件,拷来拷去都不会“环境不一致”;哪天真要上服务器,SQLite3 换成 MySQL 之类的,大不了把那几个 sqlite3.connect 换掉,其他逻辑基本不用动。
反正做完这个东西之后,老师每次见我都笑眯眯的,唯一的小问题就是,她现在看到我就说:“东哥,下次再给我加个导出 Excel 的功能哈。” 行吧,下次,下次说不定就变成“学生成绩可视化大屏系统”了……