内容概要
- python操作MySQL
- SQL注入问题
- 修改表SQL语句补充
- 视图、触发器、存储过程
- 事务
- 流程控制、函数
- 索引与慢查询优化
python操作MySQL
| python中支持操作MySQL的模块很多 其中最常见的当属'pymysql' |
| |
| pip3 install pymysql |
| |
| import pymysql |
| |
| con_obj = pymysql.connect( |
| host='127.0.0.1', |
| port=3306, |
| user='root', |
| password='123456', |
| database='lrean', |
| charset='utf8' |
| ) |
| |
| cursor = con_obj.cursor( |
| cursor=pymysql.cursors.DictCursor |
| ) |
| |
| sql = 'show tables' |
| |
| |
| |
| affect_rows = cursor.execute(sql) |
| print(affect_rows) |
| |
| res = cursor.fetchall() |
| print(res) |
| |
| '''补充说明''' |
| 获取SQL语句执行的结果 跟读取文件内容的read方法几乎一致(光标) |
| fetchone() |
| fetchmany() |
| fetchall() |
| cursor.scroll(1, 'relative') |
| cursor.scroll(1, 'absolute') |

SQL注入问题
代码
| import pymysql |
| |
| conn_obj = pymysql.connect( |
| host='127.0.0.1', |
| port=3306, |
| user='root', |
| password='123456', |
| database='lrean', |
| charset='utf8' |
| ) |
| cursor = conn_obj.cursor( |
| cursor=pymysql.cursors.DictCursor |
| ) |
| |
| name = input('请输入您的用户名>>>:').strip() |
| password = input('请输入您的密码>>>:').strip() |
| |
| sql = "select * from userinfo where name='%s' and password='%s'; " %(name,password) |
| |
| cursor.execute(sql) |
| res = cursor.fetchall() |
| if res: |
| print('登录成功') |
| else: |
| print('用户名或密码错误') |
| |
| 用户名:tom '-- hhshcm |
| 密码:直接回车 |
| # 用户名和密码都不需要也可以登录 |
| 用户名:xxx' or 1=1 -- asdjasjdkajsd |
| 密码:直接回车 |
| """上述现象就是典型的SQL注入问题""" |
| 上述情况利用的是MySQL注释语法及逻辑运算符 |
| |
| |
| execute方法自带校验SQL注入问题 自动处理特殊符号 |
| ps:设计到敏感数据的拼接 全部交给execute方法即可!!! |
| sql = "select * from userinfo where name='%s%' and password='%s';" |
| cursor.execute(sql, (name, password)) |
| |
| """ |
| execute方法补充(了解) |
| 批量插入数据 |
| sql = 'insert into userinfo(name,password) values(%s,%s)' |
| cursor.executemany(sql,[('tom',123),('lavin',321),('pony',333)]) |
| """ |
二次确认
| """ |
| 数据的增删改查四个操作是有轻重之分的 |
| 查 不会影响真正的数据 重要程度最低 |
| 增、改、删 都会影响真正的数据 重要程度较高 |
| pymysql针对增、改、删三个操作 都设置了二次确认 如果不确认则不会真正影响数据库 |
| """ |
| 方式1:代码直接编写 |
| affect_row = cursor.execute(sql) |
| conn_obj.commit() |
| 方式2:配置固定参数 |
| conn_obj = pymysql.connect( |
| autocommit=True |
| ) |
修改表SQL语句补充
| |
| alter table t1 rename ttt; |
| |
| alter table ttt add pwd int; '''默认是尾部追加字段''' |
| alter table ttt add tid int after name; '''指定追加位置''' |
| alter table ttt add nid int first; '''指定头部添加字段''' |
| |
| alter table ttt change pwd password tinyint; |
| |
| alter table ttt drop nid; |
视图
| """ |
| 视图概念 |
| 通过SQL语句的执行得到的一张虚拟表 保存下来之后就称之为'视图' |
| 视图的作用 |
| 如果需要频繁的使用一张虚拟表 可以考虑制作成视图 降低操作难度 |
| eg:emp表与dep表拼接 |
| 视图的制作 |
| create view 视图名 as sql语句 |
| """ |
| |
| create view 视图名 as sql语句 |
| |
| 查看视图的字段信息 |
| describe 视图名; |
| 查看视图的详细信息 |
| show create view 视图名; |
| |
| |

触发器
| """ |
| 触发器的概念 |
| 在对表数据进行增、删、改的操作下,自动触发的功能 |
| 触发器的作用 |
| 专门针对表数据操作 定制个性化配套功能 |
| 触发器种类 |
| 表数据新增之前、新增之后 |
| 表数据修改之前、修改之后 |
| 表数据删除之前、删除之后 |
| 触发器创建 |
| create tirgger 触发器名 before/after insert/update/delect |
| on 表名 for each row |
| begin |
| SQL语句 |
| end |
| 触发器一般要见名知义 建议采取列式布局形式 |
| tri_after_insert_t1 |
| tri_befor_update_t3 |
| tri_befor_delect_t3 |
| """ |
触发器案例(了解)
| 1.先创建两张表 |
| |
| CREATE TABLE cmd ( |
| id INT PRIMARY KEY auto_increment, |
| USER CHAR (32), |
| priv CHAR (10), |
| cmd CHAR (64), |
| sub_time datetime, |
| success enum ('yes', 'no') |
| ); |
| |
| CREATE TABLE errlog ( |
| id INT PRIMARY KEY auto_increment, |
| err_cmd CHAR (64), |
| err_time datetime |
| ); |
| 2.需求:cmd表插入数据的success如果值为no 则去errlog表中插入一条记录 |
| "delimiter $$ 将mysql默认的结束符由;换成$$" |
| delimiter $$ |
| create trigger tri_after_insert_cmd after insert on cmd for each row |
| begin |
| if NEW.success = 'no' then |
| insert into errlog(err_cmd,err_time) values(NEW.cmd,NEW.sub_time); |
| end if; |
| end $$ |
| delimiter ; |
| 3.仅仅往cmd表中插入数据 |
| INSERT INTO cmd ( |
| USER, |
| priv, |
| cmd, |
| sub_time, |
| success |
| )VALUES |
| ('kevin','0755','ls -l /etc',NOW(),'yes'), |
| ('kevin','0755','cat /etc/passwd',NOW(),'no'), |
| ('kevin','0755','useradd xxx',NOW(),'no'), |
| ('kevin','0755','ps aux',NOW(),'yes'); |
| 4.触发器其他补充 |
| 查看当前库下所有的触发器信息 |
| show triggers\G; |
| 删除当前库下指定的触发器信息 |
| drop trigger 触发器名称; |


事务
| """ |
| 事务的概念 |
| 事务可以包含诸多SQL语句并且这些SQL语句 |
| 要么同时执行成功 要么同时执行失败 这是事务的原子性特点 |
| 事务的作用 |
| 戴某某欠了赵某某一笔钱 现在想要还钱 |
| 戴某某拿着交行的银行卡去招商银行的ATM机给赵某某的建行卡转钱 |
| 1.朝交行的服务器发送请求 修改戴某某账户余额(减钱) |
| 2.朝建行的服务器发送请求 修改赵某某账户余额(加钱) |
| 事务的四大特性(重点) |
| ACID |
| A:原子性 |
| 一个事务是一个不可分割的整体 里面的操作要么都成立要么都不成立 |
| C:一致性 |
| 事务必须使数据库从一个一致性状态变到另外一个一致性状态 |
| I:隔离性 |
| 并发编程中 多个事务之间是相互隔离的 不会彼此干扰 |
| D:持久性 |
| 事务一旦提交 产生的结果应该是永久的 不可逆的 |
| 课下可以自己百度搜索整理 上面是老师整理的简单概述版本 |
| ps:辛辛苦苦一个月 换来的就是一条SQL语句!!! |
| """ |
| 具体使用 |
| 1.创建表及录入数据 |
| create table user( |
| id int primary key auto_increment, |
| name char(32), |
| balance int |
| ); |
| insert into user(name,balance) |
| values |
| ('jason',1000), |
| ('kevin',1000), |
| ('tank',1000); |
| 2.事务操作 |
| 开启一个事务操作 |
| start transaction; |
| 编写SQL语句(同属一个事务) |
| update user set balance=900 where name='jason'; |
| update user set balance=1000 where name='kevin'; |
| update user set balance=1010 where name='tank'; |
| 事务回滚(返回执行事务操作之前的数据库状态) |
| rollback; |
| 事务确认(执行完事务的主动操作之后 确认无误之后 需要执行确认命令) |
| commit; |



存储过程
| 类似于python中的自定义函数 |
| |
| |
| delimiter $$ |
| create procedure p1() |
| begin |
| select name,balance from user; |
| end $$ |
| delimiter ; |
| |
| |
| call p1() |
| |
| """ |
| 类似于有参函数 |
| delimiter $$ |
| create procedure p1( |
| in m int, # in表示这个参数必须只能是传入不能被返回出去 |
| in n int, |
| out res int # out表示这个参数可以被返回出去,还有一个inout表示即可以传入也可以被返回出去 |
| ) |
| begin |
| select tname from userinfo where id > m and id < n; |
| set res=0; # 用来标志存储过程是否执行 |
| end $$ |
| delimiter ; |
| |
| # 针对res需要先提前定义 |
| set @res=10; 定义 |
| select @res; 查看 |
| call p1(1,5,@res) 调用 |
| select @res 查看 |
| |
| 查看存储过程具体信息 |
| show create procedure pro1; |
| 查看所有存储过程 |
| show procedure status; |
| 删除存储过程 |
| drop procedure pro1; |
| """ |

函数
注意与存储过程的区别,mysql内置的函数只能在sql语句中使用!
| "ps:可以通过help 函数名 查看帮助信息!" |
| |
| Trim、LTrim、RTrim |
| |
| |
| Lower、Upper |
| |
| |
| Left、Right |
| |
| |
| Soundex |
| """ |
| eg:客户表中有一个顾客登记的用户名为J.Lee |
| 但如果这是输入错误真名其实叫J.Lie,可以使用soundex匹配发音类似的 |
| where Soundex(name)=Soundex('J.Lie') |
| """ |
| |
| |
| '''在MySQL中表示时间格式尽量采用2022-11-11形式''' |
| CREATE TABLE blog ( |
| id INT PRIMARY KEY auto_increment, |
| NAME CHAR (32), |
| sub_time datetime |
| ); |
| INSERT INTO blog (NAME, sub_time) |
| VALUES |
| ('第1篇','2015-03-01 11:31:21'), |
| ('第2篇','2015-03-11 16:31:21'), |
| ('第3篇','2016-07-01 10:21:31'), |
| ('第4篇','2016-07-22 09:23:21'), |
| ('第5篇','2016-07-23 10:11:11'), |
| ('第6篇','2016-07-25 11:21:31'), |
| ('第7篇','2017-03-01 15:33:21'), |
| ('第8篇','2017-03-01 17:32:21'), |
| ('第9篇','2017-03-01 18:31:21'); |
| |
| select date_format(sub_time,'%Y-%m'),count(id) from blog group by date_format(sub_time,'%Y-%m'); |
| |
| 1.where Date(sub_time) = '2015-03-01' |
| 2.where Year(sub_time)=2016 AND Month(sub_time)=07; |
| |
| adddate 增加一个日期 |
| addtime 增加一个时间 |
| datediff 计算两个日期差值 |
| ... |


)

流程控制
| |
| if 条件: |
| 子代码 |
| elif 条件: |
| 子代码 |
| else: |
| 子代码 |
| |
| if(条件){ |
| 子代码 |
| }else if(条件){ |
| 子代码 |
| }else{ |
| 子代码 |
| } |
| |
| if 条件 then |
| 子代码 |
| elseif 条件 then |
| 子代码 |
| else |
| 子代码 |
| end if; |
| |
| |
| DECLARE num INT ; |
| SET num = 0 ; |
| WHILE num < 10 DO |
| SELECT num ; |
| SET num = num + 1 ; |
| END WHILE ; |
索引
| |
| 类似于书的目录。意味着以后再查数据应该先找目录再找数据,而不是用翻页的方式查询数据 |
| |
| 索引在MySQL中也叫做“键”,是存储引擎用于快速找到记录的一种数据结构 |
| primary key 主键 |
| unique key 唯一键 |
| index key 索引键 |
| 上面三种key前两种除了有加速查询的效果之外还有额外的约束条件(primary key:非空且唯一,unique key:唯一),而index key没有任何约束功能只会帮你加速查询 |
| |
| |
| |
| id name pwd post_comment addr age |
| 基于id查找数据很快 但是基于addr查找数据就很慢 |
| 解决的措施可以是给addr添加索引 |
| '''索引虽然好用 但是不能无限制的创建!!!''' |
| **索引的影响:** |
| * 在表中有大量数据的前提下,创建索引速度会很慢 |
| * 在索引创建完毕后,对表的查询性能会大幅度提升,但是写的性能会降低 |
| |
| 索引的底层数据结构是b+树 |
| b树 红黑树 二叉树 b*树 b+树 |
| 上述结构都是为了更好的基于树查找到相应的数据 |
| |
| 只有叶子结点存放真实数据,根和树枝节点存的仅仅是虚拟数据 |
| 查询次数由树的层级决定,层级越低次数越少 |
| 一个磁盘块儿的大小是一定的,那也就意味着能存的数据量是一定的。如何保证树的层级最低呢?一个磁盘块儿存放占用空间比较小的数据项 |
| 思考我们应该给我们一张表里面的什么字段字段建立索引能够降低树的层级高度>>> 主键id字段 |
| |
| """ |
| 聚集索引(primary key) |
| 辅助索引(unique key,index key) |
| 查询数据的时候不可能都是用id作为筛选条件,也可能会用name,password等字段信息,那么这个时候就无法利用到聚集索引的加速查询效果。就需要给其他字段建立索引,这些索引就叫辅助索引 |
| |
| 叶子结点存放的是辅助索引字段对应的那条记录的主键的值(比如:按照name字段创建索引,那么叶子节点存放的是:{name对应的值:name所在的那条记录的主键值}) |
| 数据查找 如果一开始使用的是辅助索引 那么还需要使用聚焦索引才可以获取到真实数据 |
| |
| 覆盖索引:只在辅助索引的叶子节点中就已经找到了所有我们想要的数据 |
| select name from user where name='jason'; |
| 非覆盖索引:虽然查询的时候命中了索引字段name,但是要查的是age字段,所以还需要利用主键才去查找 |
| select age from user where name='jason'; |
| """ |

【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· winform 绘制太阳,地球,月球 运作规律
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人