当前位置:首页 » tkinter » 正文

tkinter之SQLite操作

这篇文章我来讲一下python内置的数据库sqlite3,这本是python知识,我们先是温习一下sqlite3知识,然后结果上一篇文章讲解的treeview组件或别的组件来显示数据库的增,查,改,删等操作结果。

先复习sqlite数据库知识。

1. 创建数据库文件

2. 操作SQLite: 增加记录,查看记录,修改记录,删除记录

 

对于SQLite操作的步骤:

1. 导入sqlite3模块

2. 创建Connection连接对象

3. 创建Cursor游标对象

4. 执行SQL语句,操作数据库的增,查,改,删等操作

5. 关闭游标对象

6. 增,查,改,删操作要提交事务

7. 关闭连接对象

 

好,先来创建一个数据库test1.db,并新建一个表user,表里有设2个字段:id,name,id设为主键,id字段属性为整型,占10个字节,name字段为字符型,占20个字节。下面的代码实现了以上功能:

 

创建数据库及创建一个数据表user

import sqlite3  # 导入sqlite3模块
 
# 创建连接对象
conn=sqlite3.connect('test1.db') #创建或打开数据库
 
# 创建游标对象
cur=conn.cursor() # 创建游标对象
 
# SQL语句
cur.execute('create table user (id int(10) primary key, name varchar(20))') 
# 创建表:user,2个字段:id,name,id为主键
 
# 关闭游标对象
cur.close()
 
# 提交事务
# --- 数据库的增,改,删要提交事务
 
# 关闭连接对象
conn.close()


运行后,就在你的代码目录里生成test1.db , 里面有一个表user,大家可以用Navicat for SQLite 等相关软件查看这个数据库里的user表是否生成。

 

如果你再次运行上面的代码,会出错,因为user表已经生成,不能再生成了。

好,我就在这个表的基础上,进行增加,查找,修改,删除的操作。

 

先增加几条记录,增加记录,也是操作SQL操作。操作数据库的步骤就像我前面说的几个步骤。

下面的代码会增加一条记录:1,‘刘一’

 

数据表增加一条记录:

import sqlite3  # 导入sqlite3模块
 
# 创建连接对象
conn=sqlite3.connect('test1.db') #创建或打开数据库
 
# 创建游标对象
cur=conn.cursor() # 创建游标对象
 
# SQL语句  # 源码来自wb98.com
# cur.execute("insert into user (id,name) values (1,'刘一')")
# 为了sql注入,不推荐上面这种方式
cur.execute('insert into user (id,name) values (?,?)',(1,'刘一'))
# 表user增加一条记录
 
# 关闭游标对象
cur.close()
 
# 提交事务
conn.commit() # 提交事务
# --- 数据库的增,改,删要提交事务
 
# 关闭连接对象
conn.close()


运行后,user表会增加一条记录。

注意,在sql语句中,为了防止sql注入风险,采用采用占用符的方式。

 

以上采用的是游标对象的execute方法新增一条记录,下代码采用executemany() 一次就新增几条记录,只要稍稍改动上面的代码就行了。    丶丌皛

 

数据表记录批量增加:

import sqlite3  # 导入sqlite3模块
 
# 创建连接对象
conn=sqlite3.connect('test1.db') #创建或打开数据库
 
# 创建游标对象
cur=conn.cursor() # 创建游标对象
 
# SQL语句
newlist=[(2,'陈二'),(3,'张三'),(4,'李四'),(5,'王五'),(6,'赵六'),
   (7,'孙七'),(8,'周八'),(9,'吴九'),(10,'郑十')]
cur.executemany('insert into user (id,name) values (?,?)',newlist)
# 表user批量增加几条记录
 
# 关闭游标对象
cur.close()
 
# 提交事务
conn.commit() # 提交事务
# --- 数据库的增,改,删要提交事务
 
# 关闭连接对象
conn.close()

以上代码,我在win10,python3.8版测试成功,Navicat for SQLite都可以打开数据库查看证实成功,但是我们现在要自己写代码输出这个数据库的内容。

  

数据表中的记录查看:

游标对象有几个方法跟记录查看有关,我们来学习或回顾一下:

方法

说明

fetchone()

获取查询结果集中的下一行,返回一个单一的序列,当没有数据时,返回 None

fetchmany(size)

获取查询结果集中的下一行组,返回一个列表。当没有更多的可用的行时,则返回一个空的列表。

fetchall()

获取查询结果集中所有(剩余)的行,返回一个列表。当没有可用的行时,则返回一个空的列表

 

下面的代码演示了上述3个方法:

import sqlite3  # 导入sqlite3模块
 
# 创建连接对象
conn=sqlite3.connect('test1.db') #创建或打开数据库
 
# 创建游标对象
cur=conn.cursor() # 创建游标对象
 
# SQL语句
cur.execute('select * from user where id=?',(1,)) #  查询第一条记录
result1=cur.fetchone() # 获取一条查询记录
print('第1条记录:',result1)
print('-----')
 
cur.execute('select * from user where id between ? and ?',(1,5)) # 查询id值范围为1-5之间的记录
result2=cur.fetchmany(5) # 获取5条查询记录
print('id=1至5的记录:',result2)
print('-----')
 
cur.execute('select * from user') # 查询所有记录
result3=cur.fetchall() # 获取所有的查询记录
print('所有的记录:',result3)
print('-----')
 
# 关闭游标对象
cur.close()
 
# 提交事务
conn.commit() # 提交事务
# --- 数据库的增,改,删要提交事务
 
# 关闭连接对象
conn.close()

运行代码,输出:

第1条记录: (1, '刘一')

-----

id=1至5的记录: [(1, '刘一'), (2, '陈二'), (3, '张三'), (4, '李四'), (5, '王五')]

-----

所有的记录: [(3, '张三'), (4, '李四'), (5, '王五'), (6, '赵六'), (7, '孙七'), (8, '周八'), (9, '吴九'), (10, '郑十')]

-----

 

数据库的记录修改:

数据表中的记录修改,我们也是通过execute()方法来操作sql语句来达成目的。下面的代码来演示一下,把第一条记录的name改成“曹操”。

import sqlite3  # 导入sqlite3模块
 
# 创建连接对象
conn=sqlite3.connect('test1.db') #创建或打开数据库
 
# 创建游标对象
cur=conn.cursor() # 创建游标对象
 
# SQL语句
cur.execute('update user set name=? where id=?',('曹操','1'))
# 修改id=1的记录 name='曹操'
 
# 关闭游标对象
cur.close()
 
# 提交事务
conn.commit() # 提交事务
# --- 数据库的增,改,删要提交事务
 
# 关闭连接对象
conn.close()


 数据表中的记录删除:

记录的删除,当然也是通过execute()方法来操作sql语句来达成目的。只要把上面的SQL语句改一下就行,其它都不用改变。

import sqlite3  # 导入sqlite3模块
 
# 创建连接对象
conn=sqlite3.connect('test1.db') #创建或打开数据库
 
# 创建游标对象
cur=conn.cursor() # 创建游标对象
 
# SQL语句
cur.execute('delete from user where id=?',(10,))
# 删除id=10的记录
 
# 关闭游标对象
cur.close()
 
# 提交事务
conn.commit() # 提交事务
# --- 数据库的增,改,删要提交事务
 
# 关闭连接对象
conn.close()

小醒一下:数据库的记录的增加,修改,删除要提交事务才可以更新成功。   丶丌皛

 

       上面是对SQLite3操作的基本知识的学习或回顾,下面我们要利用tkinter学习的组件知识来显示对数据库的操作结果。

Treeview组件用来显示数据表是非常合适的,下面就来讲一下如何显示数据表中的记录。

  

下面代码,是在上面的代码基础上,再加入创建一个treeview组件,用循环的方法,将fetchall() 返回的所有记录列表中的元素,一一插入到treeview组件里。这个难度并不大。

from tkinter import *
from tkinter.ttk import *
import sqlite3 # 导入sqlite3模块
 
root=Tk() # 源码来自wb98.com
 
# 创建连接对象
conn=sqlite3.connect('test1.db') #创建或打开数据库
 
# 创建游标对象
cur=conn.cursor() # 创建游标对象
 
# SQL语句
cur.execute('select * from user')
result=cur.fetchall() # 返回所有记录列表
 
# 关闭游标对象
cur.close()
 
# 提交事务
# --- 只是查询,不用提交事务
 
# 关闭连接对象
conn.close()
 
tree1=Treeview(root,columns=('c1','c2'),show='headings') # 创建无图标栏的单列表栏
tree1.column('#1',width=80,anchor=CENTER) # 设置组件标题栏宽度,列对齐方式
tree1.column('#2',width=80,anchor=CENTER)
tree1.heading('c1',text='ID') # 设置组件标题栏文本
tree1.heading('c2',text='姓名')
 
for i in result: # 循环把列表元素一一加入tree1组件
    tree1.insert('',index=END,values=i)
 
tree1.pack()
 
root.mainloop()

运行结果

             1.PNG

  

上面的数据库稍简单一点,不好演示SQL语句,我用代码,由一个文本导入到数据库。

先在test1.db这个数据库,再新建一个表student,然后用行方法读取文本db.txt,把一行作为一行记录读取到student表中。

下面是db.txt的内容:

1,a,3,李明,88,79,67,

2,b,5,王小梅,56,99,87,

3,a,2,陈小明,68,77,56,

4,b,6,张梅芳,74,86,76,

5,a,10,郑明,66,68,100,

6,b,8,何振华,77,87,91,

7,a,11,吴小芳,97,72,67,

8,b,1,姚明,68,77,87,

9,a,12,孙文正,89,81,73,

10,b,4,陈生国,50,86,93,

11,a,1,何云峰,88,99,96,

12,a,13,李强民,77,49,89,

13,a,4,明正,72,89,67,

14,b,3,曾红梅,48,55,86,

15,a,8,吴小京,82,67,69,

16,a,6,黄杰伦,75,74,96,

17,b,9,周春雨,95,65,99,

18,b,10,胡歌,91,99,67,

19,a,5,何英杰,89,85,91,

20,b,7,孙娜娜,79,83,99,

21,b,11,陈清锋,89,94,88,

22,a,9,李强,68,69,66,

23,b,2,何雨琴,67,93,67,

24,b,12,陈克民,98,67,56,

25,a,7,杨明明,46,100,87,

26,b,13,周玉琴,75,45,88,

 

 以下是新建数据表 student 的代码

import sqlite3  # 导入sqlite3模块
 
# 创建连接对象
conn=sqlite3.connect('test1.db') #创建或打开数据库
 
# 创建游标对象
cur=conn.cursor() # 创建游标对象
 
# SQL语句
cur.execute('''create table student (
   id INTEGER(10) primary key, 
   bj TEXT(10),
   xh TEXT(10),
   xm TEXT(10),
   yw REAL(10),
   sx REAL(10),
   yy REAL(10)
   )''') 
# 创建表student: id  bj班级 xh学号 xm姓名 yw语文 sx数学 yy英语
 
# 关闭游标对象
cur.close()
 
# 提交事务
# --- 数据库的增,改,删要提交事务
 
# 关闭连接对象
conn.close()

 

写代码把db.txt的文本导入到数据表student中

import sqlite3  # 导入sqlite3模块
 
with open('db.txt', 'r', encoding='utf-8') as file:  # 只读方式打开编码为utf-8的文本文件
    line = file.readlines()  # 以读取一行为列表方法读取全部行,line为分解好的列表内容
 
# str1[0]:id  str1[1]:bj班级  str1[2]:xh学号  str1[3]:xm姓名  str1[4]:yw语文
# str1[5]:sx数学  str1[6]:yy英语
 
# 创建连接对象
conn=sqlite3.connect('test1.db') #创建或打开数据库
 
# 创建游标对象
cur=conn.cursor() # 创建游标对象
 
for i in range(len(line)): # 以循环方法把读取的每一行插入到数据表的记录中
    str1 = line[i].split(',')  # 读取新一行记录
    cur.execute('insert into student (id,bj,xh,xm,yw,sx,yy) values (?,?,?,?,?,?,?)',str1)
 
# 关闭游标对象
cur.close()
 
# 提交事务
conn.commit() # 提交事务
# --- 数据库的增,改,删要提交事务
 
# 关闭连接对象
conn.close()

 

运行后,用Navicat for SQLite 查看test1中数据表student的结果

2.PNG

 

我就用这个数据表的数据来演示,用treeview等组件来显示数据,并顺便温习一下SQL语句知识。

以下是全部代码:

from tkinter import *
from tkinter.ttk import *
import sqlite3 # 导入sqlite3模块
 
def selectjob(strSQL):
    # 创建连接对象
    conn=sqlite3.connect('test1.db') # 创建或打开数据库
 
    # 创建游标对象
    cur=conn.cursor() # 创建游标对象
 
    for i in tree1.get_children(): # 清空treeview所有记录
        tree1.delete(i)
 
    # SQL语句
    cur.execute(strSQL)
 
    result=cur.fetchall() # 返回所有记录列表
 
    root.title(str(len(result))+"条记录")
 
    for i in result: # 循环把列表元素一一加入tree1组件
        tree1.insert('',index=END,values=i)
 
    # 关闭游标对象
    cur.close()
 
    # 提交事务
    # --- 只是查询,不用提交事务
 
    # 关闭连接对象
    conn.close()
 
root=Tk()
 
tree1=Treeview(root,columns=('c1','c2','c3','c4','c5','c6','c7'),show='headings') # 创建无图标栏的单列表栏
tree1.column('#1',width=40) # 设置组件标题栏宽度,列对齐方式
tree1.column('#2',width=40)
tree1.column('#3',width=40)
tree1.column('#4',width=80)
tree1.column('#5',width=80)
tree1.column('#6',width=80)
tree1.column('#7',width=80)
 
tree1.heading('c1',text='ID') # 设置组件标题栏文本
tree1.heading('c2',text='班级')
tree1.heading('c3',text='学号')
tree1.heading('c4',text='姓名')
tree1.heading('c5',text='语文')
tree1.heading('c6',text='数学')
tree1.heading('c7',text='英文')
 
root.rowconfigure(0,weight=1) # 某行 限制最小高度为30像素
root.columnconfigure(0,weight=1) # 某列 限制最小宽度为66像素
root.columnconfigure(1,weight=1) # 某列 限制最小宽度为66像素
root.columnconfigure(2,weight=1) # 某列 限制最小宽度为66像素
 
tree1.grid(row=0,column=0,columnspan=3,sticky=NSEW)
 
scr1=Scrollbar(root)  # 垂直滚动条
scr1.grid(row=0,column=3,sticky=N+S)
 
tree1.config(yscrollcommand = scr1.set) # Treeview和滚动条互相绑定
scr1.config(command = tree1.yview)
 
var=StringVar()
list1=('显示所有同学',
       '只显示a班同学',
       '姓名含“明”字的同学',
       '姓名第2个字是“明”的同学',
       '二字姓名,第2个字是“明”的同学',
       '3门功课有一门为100分的同学',
       '3门功课都不及格的同学',
       '学号为1,5,7,8的同学',
       '三门功课平均分>=90分的同学',
       'a班按语文成绩降序排列,辅之以数学降序,英文升序排列',
       '名字不包含“雨”字的同学',
       '语文分数在90到100之间高分同学',
       '语文分数在60到90分范围以外的高分和低分同学,降序排列',
       '姓名第2字和第3字相同的姓名',
       '显示姓“郑,何,周”的同学',
       '显示本文章的作者的姓名:何云峰(wb98·com)',
       )
com1=Combobox(root,textvariable=var,value=list1,state='readonly')
com1.grid(row=1,column=0,columnspan=4,sticky=W+E)
com1.current(0)
 
def com1selection(event):
    # print('选择后:',com1.get())
    print('选择项序号:',com1.current())
    index=com1.current() # 选择项序号
    if index==0: # 显示所有同学
        stra='select * from student'
    elif index==1: # 只显示a班同学
        stra='select * from student where bj="a"'
    elif index==2: # 姓名含“明”字的同学
        stra='select * from student where xm like "%明%"'
    elif index==3: # 姓名第2个字是“明”的同学
        stra='select * from student where xm like "_明%"'
    elif index==4: # 二字姓名,第2个字是“明”的同学
        stra='select * from student where xm like "_明"'
    elif index==5: # 3门功课有一门为100分的同学
        stra='select * from student where yw="100" or sx="100" or yy="100"'
    elif index==6: # 3门功课都不及格的同学
        stra='select * from student where yw<"60" and sx<"60" or yy<"60"'
    elif index==7: # 学号为1,5,7,8的同学
        stra='select * from student where xh in (1,5,7,8)'
    elif index==8: # 三门功课平均分>=90分的同学
        stra='select * from student where (yw+sx+yy)/3>=90'
    elif index==9: # a班按语文成绩排序(降序),辅之以数学降序,英文升序排列
        stra='select * from student  where bj="a" order by yw desc,sx desc,yy asc'
    elif index==10: # 名字不包含“雨”字的同学
        stra='select * from student where xm not like "%雨%"'
    elif index==11: # 语文分数在90到100之间的同学
        stra='select * from student where yw between 90 and 100'
    elif index==12: # 语文分数在60到90分范围以外的高分和低分同学,降序排列
        stra='select * from student where yw not between 60 and 90 order by yw'
    elif index==13: # 姓名第2字和第3字相同的姓名
        stra='select * from student where substr(xm,2,1)=substr(xm,3,1)'
    elif index==14: # 显示姓“郑,何,周”的同学
        stra=stra='select *  from student where substr(xm,1,1) in ("郑","何","周")'
    elif index==15: # 显示本文章的作者的姓名:何云峰(wb98·com)
        stra=stra='select *  from student where xm="何云峰"'
 
    selectjob(stra)
 
com1.bind("<<ComboboxSelected>>",com1selection)
 
root.mainloop()

运行结果


sqlite3数据表的记录在treeview组件的显示及sql语句知识,我就演示到这里。

包括这篇文章在内,我也写的文章都是我学习tkinter编程知识的笔记,以后更多的是写一些实际的小程序代码,加深对知识的了解,巩固学过的知识,你想跟我一起学习tkinter,ttk编程吗?请关注我。收藏我的文章,谢谢。我的网站 wb98.com

 


打赏 支付宝打赏 微信打赏

来源:济亨网

本文链接:https://wb98.com/post/343.html

    << 上一篇 下一篇 >>

    湘公网安备 43011102000514号 - 湘ICP备08100508号