Python | GUI小项目_数据库批量处理器1.0

平时的 Python 学习过程比较枯燥,想要保持兴趣和提高,就得上手亲自做一些小项目
本次小项目是利用 Python 自带的 tkinter 模块构造简单的 GUI,利用PyMySQL 对数据库进行操作 ,数据存储利用 json模块,这些都是比较基础的东西,做的过程可以逐渐熟悉上手

此工具可应用于多个数据库存在相同的表名等情况下,进行批量操作

开发环境如下:

  • Python 3.6.6
  • MySQL 8.0
  • PyMySQL 0.9.2 (pip即可)
  • IDE:Geany

先来看看项目完成的界面

主界面

主界面.PNG

添加键界面

添加界面.PNG

功能说明
1、左上方是 SQL 语句输入栏,输入语句后点执行按键,执行按键会从本地存储的 json 文件获取数据库信息,连接数据库,然后执行语句,再关闭数据库
2、主界面有 2 个清屏按键,分别为语句清屏和 Info 清屏;语句清屏,顾名思义是将 SQL 语句清除, Info 清屏是将下方的执行信息栏的报错清空
3、退出程序按键,退出该程序
4、添加按键,点击后弹出添加界面,有 6 个输入栏,其中一个带有默认值, 6 栏全部输入信息才能点击确认,否则出现提示框提示将信息填写完整,取消即退出该界面;点击确认后,新的数据库信息将保存到本地 json 文件中,同时将已保存的数据库信息清空,提示更新键更新
5、删除按键,点击后,删除已保存的数据库框中选中的数据库,同时本地 json 文件会删除相应的信息
6、更新按键,点击后,获取本地 json 文件数据,同时将数据更新到上方的框中
7、执行信息栏,在程序报错时,栏中将出现相对应的报错信息

!由于是练手项目,基本功能不影响的情况存在未知BUG,以后在实际应用中修改BUG,再重构优化

话不多说,根据实际开发过程中,下面依开发思路进行分解

1、先根据需求对GUI界面进行分解、排版

  • 需要3个标签对信息框介绍——3个Label
  • 需要7个按键实现不同的功能——7个Button
  • 需要3个框,分别用来输入SQL语句、展示数据库信息、展示操作中发生的必要信息——2个scrolledtext、1个Listbox

我的布局图

布局.png

PS:实际可以更简单,不需要分这么多块,但也是第一开发,经验不足,Frame可有可无,方便分块,看实际情况

此时应有

from tkinter import *
from tkinter import scrolledtext
def main():
    root = Tk()

    # Frame
    frmA1 = Frame()
    frmA2 = Frame()
    frmA3 = Frame()
    frmB1 = Frame()
    frmB2 = Frame()
    frmB3 = Frame()
    frmC =  Frame()

    # Label
    label_1 = Label(frmA1)
    label_2 = Label(frmB1)
    label_3 = Label(frmC)

    # scrolledtext
    scrt_1 = scrolledtext.ScrolledText(frmA2)
    scrt_2 = scrolledtext.ScrolledText(frmC)

    # Listbox
    scrollbar = Scrollbar(frmB2)
    listbox = Listbox(frmB2)

    # Button
    button_1 = Button(frmA3)
    button_2 = Button(frmA3)
    button_3 = Button(frmA3)
    button_4 = Button(frmB3)
    button_5 = Button(frmB3)
    button_6 = Button(frmB3)
    button_7 = Button(frmC)

    # 窗口布局
    frmA1.grid()
    frmA2.grid()
    frmA3.grid()
    frmB1.grid()
    frmB2.grid()
    frmB3.grid()
    frmC.grid()
        
    label_1.grid()
    label_2.grid()
    label_3.grid()
        
    scrt_1.grid()
    scrt_2.grid()
    
    scrollbar.grid()
    listbox.grid()

    button_1.grid()
    button_2.grid()
    button_3.grid()
    button_4.grid()
    button_5.grid()
    button_6.grid()
    button_7.grid()

    root.mainloop()

if __name__ == '__main__':
    main()

将所有需要的控件写出来后,需要对各个控件进行重命名、属性的添加,布局的调整,宽度和高度没有设置好以及grid调用时,一般都会出现不整齐的现象,这些都要需要花时间去慢慢调整,下面是完成后的代码

    # 主界面
    root = Tk()
    root.title('数据库批量处理器')
    root.resizable(0, 0)

    # Frame
    frm_width = 380
    frm_height1 = 200
    frm_height2 = 50
    frmA1 = Frame(width=frm_width, height=frm_height2-5)
    frmA2 = Frame(width=frm_width, height=frm_height1)
    frmA3 = Frame(width=frm_width, height=frm_height2)
    frmB1 = Frame(width=frm_width, height=frm_height2-5)
    frmB2 = Frame(width=frm_width, height=frm_height1)
    frmB3 = Frame(width=frm_width, height=frm_height2)
    frmC =  Frame(width=frm_width*2, height=frm_height1+30)

    # Label
    infotext = '执' + 18*' ' + '行' + 18*' ' + '信' + 18*' ' + '息' + 18*' ' + '栏'
    input_label = Label(frmA1, text='SQL   语   句   输   入   栏', width=54, height=2, anchor='center', relief='groove', foreground='red', bd=3)
    db_label =    Label(frmB1, text='已   保   存   的   数   据   库', width=54, height=2, anchor='center', relief='groove', foreground='red', bd=3)
    info_label =  Label(frmC, text=infotext, width=90, height=2, foreground='red', relief='groove', bd=3)

    # scrolledtext
    input_text = scrolledtext.ScrolledText(frmA2, width=51, height=15, wrap=WORD)
    info_text =  scrolledtext.ScrolledText(frmC, width=105, height=13, wrap=WORD, state='normal')
    
    # Listbox
    scr = Scrollbar(frmB2, width=16)
    db_list = Listbox(frmB2, width=51, height=11, yscrollcommand=scr.set)
    scr.config(command=db_list.yview)
    
    # Button
    button_width = 16
    button_height = 2
    exe_button =    Button(frmA3, text='执 行', width=button_width, height=button_height, foreground='blue', command=exe_input, relief='groove')
    clear1_button = Button(frmA3, text='语 句 清 屏', width=button_width, height=button_height, foreground='blue', command=clear_input1, relief='groove')
    quit_button =   Button(frmA3, text='退 出 程 序', width=button_width, height=button_height, foreground='blue', command=quitcommand, relief='groove')
    add_button =    Button(frmB3, text='添 加', width=button_width, height=button_height, foreground='blue', command=create_add_frame, relief='groove')
    del_button =    Button(frmB3, text='删 除', width=button_width, height=button_height, foreground='blue', relief='groove', command=del_db)
    update_button = Button(frmB3, text='更 新', width=button_width, height=button_height, foreground='blue', relief='groove', command=update_db_info)
    clear2_button = Button(frmC, text='Info 清 屏', width=button_width, height=button_height, foreground='blue', command=clear_input2, relief='groove')

    # 窗口布局
    frmA1.grid(row=0, column=0)
    frmA2.grid(row=1, column=0)
    frmA3.grid(row=2, column=0)
    frmB1.grid(row=0, column=1)
    frmB2.grid(row=1, column=1)
    frmB3.grid(row=2, column=1)
    frmC.grid(row=3, column=0, columnspan=3)

    input_label.grid(row=0, column=0, padx=1, pady=1)
    db_label.grid(row=0, column=1, padx=1, pady=1)
    info_label.grid(row=0, column=0)
    
    input_text.grid(row=0, column=0)
    info_text.grid(row=1, columnspan=2)
    
    db_list.grid(row=0, column=0)
    scr.grid(row=0, column=1, ipady=80)
    
    exe_button.grid(row=0, column=0, ipadx=1, pady=1)
    clear1_button.grid(row=0, column=1, padx=1, pady=1)
    quit_button.grid(row=0, column=2, padx=1, pady=1)
    add_button.grid(row=0, column=0, padx=1, pady=1)
    del_button.grid(row=0, column=1, padx=1, pady=1)
    update_button.grid(row=0, column=2, padx=1, pady=1)
    clear2_button.grid(row=0, column=1, pady=1)

2、对各Button的功能实现

    # 主窗口方法
    # 获取本地json数据
    json_data = []
    def get_json_data():
        try:
            with open('DB_data.json') as f:
                for line in f:
                    json_data.append(json.loads(line))
            return json_data
        except:
            info_text.insert(END, 'Error: 请检查是否有添加的数据库信息及文件!\n')
    
    # 退出主界面
    def quitcommand():
        root.quit()
        
    # 清空语句  
    def clear_input1():
        input_text.delete('0.0', END)
    
    # 清空信息
    def clear_input2():
        info_text.delete('0.0', END)
    
    # 删除listbox中选中的数据库信息,以及本地数据库信息文件的更改
    def del_db():
        index = db_list.curselection()[0]
        db_list.delete(index)
        with open('DB_data.json', 'r') as f:
            dataList = f.readlines()
            del(dataList[index])
        with open('DB_data.json', 'w') as f_w:
            for line in dataList:
                f_w.write(line)
    
    # 获取json文件进行信息的更新
    def update_db_info():
        get_json_data()
        if not db_list.size():
            for i in range(len(json_data)):
                string = '第' + str(i+1) + '个数据库' + '  ------  ' + 'DBname:  ' + json_data[i]['DBname'] + '  ------  ' +'Port:  ' + json_data[i]['port'] + '\n'
                db_list.insert(END, string)
        
    # 获取json文件数据库的信息,连接,执行语句,关闭连接
    def exe_input():
        get_json_data()
        for data in json_data:
            sql = input_text.get('0.0', END)
            try:
                conn = pymysql.connect(host=data['host'], user=data['user'], password=data['password'], port=int(data['port']), db=data['DBname'], charset=data['charset'])
                cursor = conn.cursor()
                cursor.execute(sql)
                conn.commit()
                info_text.insert(END, '操作完成')
            except Exception as e:
                conn.rollback()
                error_info = ('发生错误:' + str(e) + '\n')
                info_text.insert(END, error_info)
            finally:
                cursor.close()
                conn.close()

这里我们需要了解Python操作数据库的方法,以及文件操作的知识、json的基本使用,因此我们还要import相应的包
PS:
python 2.7 使用 MySQLdb 进行数据库操作
python 3.x 使用 pymysql 进行数据库操作

3、添加界面实现

    # 创建添加二级窗口
    def create_add_frame():
        # 二级窗口方法
        def cancelCommand():
            add.destroy()
        
        def confirmCommand():
            data = {'host':host_entry.get(), 'user':user_entry.get(), 'password':ps_entry.get(), 'port':port_entry.get(), 'DBname':db_entry.get(), 'charset':charset_entry.get()}
            count = 0
            for v in data.values():
                if v != '':
                    count += 1
            if count == 6:
                with open('DB_data.json', 'a', encoding='utf-8') as f:
                    json.dump(data, f)
                    f.writelines('\n')
                    messagebox.showinfo('已完成', '数据库信息已添加!保存的数据将清空,请点击更新键更新!')
                    db_list.delete(0,END)
                    add.destroy()
            else:
                messagebox.showinfo('提示', '请将6个参数填写完整!')
        
        add = Toplevel() 
        add.title('添加新的数据库')
        add.resizable(0, 0)
        add.wm_attributes('-topmost',1)  # 将add窗口置于root前
        
        # Label
        label_width = 20
        label_height = 1
        host_label =    Label(add, text='Host:', width=label_width, height=label_height)
        user_label =    Label(add, text='User:', width=label_width, height=label_height)
        ps_label =      Label(add, text='Password:', width=label_width, height=label_height)
        port_label =    Label(add, text='Port:', width=label_width, height=label_height)
        db_label =      Label(add, text='DBname:', width=label_width, height=label_height)
        charset_label = Label(add, text='Charset:', width=label_width, height=label_height)
        
        # Entry
        entry_width = 30
        host_entry =    Entry(add, width=entry_width, bd=3)
        user_entry =    Entry(add, width=entry_width, bd=3)
        ps_entry =      Entry(add, width=entry_width, bd=3)
        port_entry =    Entry(add, width=entry_width, bd=3)
        db_entry =      Entry(add, width=entry_width, bd=3)
        charset_entry = Entry(add, width=entry_width, bd=3)
        host_entry.focus()
        charset_entry.insert(END, 'utf8')
        
        # Button
        bt_width = 10
        bt_height = 2
        confirm_button = Button(add, text='确认', width=bt_width, height=bt_height, foreground='blue', command=confirmCommand)
        cancel_button =  Button(add, text='取消', width=bt_width, height=bt_height, foreground='blue', command=cancelCommand)
        
        # 窗口布局
        host_label.grid(row=0, column=0)
        user_label.grid(row=1, column=0)
        ps_label.grid(row=2, column=0)
        port_label.grid(row=3, column=0)
        db_label.grid(row=4, column=0)
        charset_label.grid(row=5, column=0)
        
        host_entry.grid(row=0, column=1)
        user_entry.grid(row=1, column=1)
        ps_entry.grid(row=2, column=1)
        port_entry.grid(row=3, column=1)
        db_entry.grid(row=4, column=1)
        charset_entry.grid(row=5, column=1)
        
        confirm_button.grid(row=6, column=0)
        cancel_button.grid(row=6, column=1)
        

添加界面会出现messagebox的使用,需要import,各功能都测试过使用正常,出现的BUG基本不影响使用,编码时间2天完成,以后如有时间,将对此工具进行改进

各部分代码全都展示完整,如果对pymysql和json的基本操作不熟,可以继续往下看,对代码中细节有疑议或有更好方法的、需要源码练习的,请留言,相信我们会交谈甚欢 φ(≧ω≦*)

1、pymysql的基本使用

import pymysql

# 创建连接,除了port是int类型,其他都是str类型
conn = pymysql.connect(host=localhost, user=root, password='123456', port=3306, db=spiders, charset='utf8')

# sql语句
sql = 'SELECT acctid, money FROM bank'
    
try:
    # 创建游标
    cursor = conn.cursor()

    # 执行sql语句
    cursor.execute(sql)

    # 对于增删改操作,需要此方法传递数据变化
    conn.commit()
except:
    # 如果执行出错,需要对数据回滚,回复原样
    conn.rollback()
finally:
    # 关闭数据库连接,否则占用后台资源
    cursor.close()
    conn.close()

2、json的基本用法

json虽然易用,但由于第一次使用,还是踩了不少坑

刚开始我是这样保存数据的

with open('x.json', 'a', encoding='utf-8') as f:
        json.dump(data, f, indent=4)

利用文本形式打开,是这样

{
    'a':'a',
    'b':'b'
}

2个数据时

{
    'a':'a',
    'b':'b'
}{
    'c':'c'.
    'd':'d'
}

是不是看着很整齐? 这都是indent=4的功劳

取出数据

with open('x.json', 'r', encoding='utf-8') as f:
    json.load(f)

你会发现,x.json中只有一个数据时,运行良好,不会报错,但存在多个时,就会出现 extra data 的报错

json进行文件操作有4个方法
1、dump()
2、dumps()
3、load()
4、loads()

有什么区别吗?
1、dump() 将dict转为str形式,并保存到文件中
2、dumps() 将dict转为str形式
3、load() 将str转为dict形式,并从文件中取出
4、loads() 将str转为dict形式

使用dumps
a = {"123":"123"}
json.dumps(a) ——> '{"123":"123"}'

使用loads
b = '{"123":"123"}'
json.loads(b) ——> {"123":"123"}

所以当json文件存在多个数据时,它无法进行转换,你可以简单认为json.load()只能对只拥有一个数据的json文件进行处理,那存在多个数据json文件就无法操作了?聪明的你已经想到先对数据择行分别取出,并分别进行转换

但这样的你要怎么分行处理呢?

{
    'a':'a',
    'b':'b'
}{
    'c':'c'.
    'd':'d'
}

第一行就一个

这样格式化的数据是因为我们设置了indent=4,如果我们没有设置这样的参数,实际的保存效果是这样的:

{"host": "localhost", "user": "root", "password": "123456", "port": "3306", "DBname": "spiders", "charset": "utf8"}

因此,当我们需要存取多个json数据时,一定不要设置indent参数,并且在存入时加入\n进行分行

所以最后的获取函数应该这样写

data = []
with open('x.json') as f:
    for line in f:
        data.append(json.loads(line))

Thank You For Watching...

小知识:如果想运行时想不出现cmd窗口,可以将文件名x.py改成x.pyw哦

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

推荐阅读更多精彩内容