mysql与python交互
MySQL 官方驱动模块 mysql connector 是mysql官方的驱动模块,兼容性特别好 下载地址:https://dev.mysql.com/downloads/connector/python/
直接查询数据库结果
mysql> show databases;
+
| Database |
+
| gpc |
| information_schema |
| mysql |
| performance_schema |
| sys |
| test |
+
6 rows in set (0.00 sec)
mysql> use test;
Database changed
mysql> show tables;
+
| Tables_in_test |
+
| gongzi |
+
1 row in set (0.00 sec)
mysql> select * from gongzi;
+
| id | name | sex | tel | years | wages |
+
| 1 | zhangsan | man | 18398989898 | 1 | 1000 |
| 2 | lisi | man | 18398555555 | 2 | 3000 |
| 3 | wangwu | man | 18398454542 | 3 | 2000 |
| 4 | mazi | man | 18398989898 | 4 | 4000 |
| 5 | xiao_ming | man | 18398989898 | 5 | 5000 |
+
5 rows in set (0.00 sec)
mysql>
使用python进行交互
import mysql.connector
con =mysql.connector.connect(
host="localhost",port = "3306",
user = "root",password = "cheng.123456789",
database = "test"
)
cursor=con.cursor()
sql="select * from gongzi;"
cursor.execute(sql)
for one in cursor:
print(one)
con.close()
data:image/s3,"s3://crabby-images/ca2b2/ca2b29f40ef298c4036ebc69485115424a4cffd8" alt="在这里插入图片描述"
sql注入与防御
data:image/s3,"s3://crabby-images/0e6b5/0e6b5d340948bf0eb8dfbe22cb6332c18b307a59" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/58928/589288ee88928fea914915850380ccded0b50986" alt="在这里插入图片描述"
import mysql.connector
config={
"host":"localhost",
"port":3306,
"user":"root",
"password":"cheng.12345",
"database":"gpc"
}
con=mysql.connector.connect(**config)
username="1 OR 1=1"
password="1 OR 1=1"
sql="SELECT COUNT(*) FROM t_user WHERE username="+username+\
" AND AES_DECRYPT(UNHEX(password),'HelloWorld')="+password;
cursor=con.cursor()
cursor.execute(sql)
print(cursor.fetchone()[0])
con.close()
data:image/s3,"s3://crabby-images/bb3a6/bb3a6859f9c0dadf3d4d4062b525c10aa0a70123" alt="在这里插入图片描述" sql注入攻击的危害:
- 由于sql语句是解释型语言,所以在拼接sql语句的时候,容易被注入恶意的sql语句
- id = “1 or 1=1”
- sql = delete from t_news where id=1 or 1=1;
mysql> select * from t_student;
+
| no | name | sex | age | email |
+
| 1 | 2 | 2 | 2 | 2 |
+
1 row in set (0.00 sec)
mysql> select * from t_news;
+
| id | title | editor_id | type_id | content_id | is_top | create_time | update_time | state |
+
| 1 | xiaoming | 121 | 123 | 2112 | 123 | 2022-05-10 21:14:38 | 2022-05-18 21:14:42 | 鑽夌? |
+
1 row in set (0.00 sec)
mysql> delete from t_news where id=1 or 1=1;
Query OK, 1 row affected (0.06 sec)
mysql> delete from t_student where no=1 or 1=1;
Query OK, 1 row affected (0.04 sec)
mysql> select * from t_news;
Empty set (0.00 sec)
mysql> select * from t_student;
Empty set (0.00 sec)
mysql>
防御
- sql 预编译机制
- 预编译sql就是数据库提前把sql语句编译成二进制,这样反复执行同一条sql语句的效率就会提升
data:image/s3,"s3://crabby-images/0ebd2/0ebd25e17826a99ccf7c237df3cfb90346257a51" alt="在这里插入图片描述" - 使用%s作为占位符代替具体的数据,传给数据库编译成二进制
data:image/s3,"s3://crabby-images/47d4d/47d4ddd4cf29f314a33cf750934df34e8e148631" alt="在这里插入图片描述" - sql语句编译的过程中,关键字已经被解析过了,所以向编译后的sql语句传入参数,都被当作字符串处理,数据库不会解析其中注入的sql语句
data:image/s3,"s3://crabby-images/54c80/54c807d1bd397c73db782d07d5bbd7d19ac0ce6c" alt="在这里插入图片描述"
import mysql.connector
config={
"host":"localhost",
"port":3306,
"user":"root",
"password":"cheng.12345",
"database":"gpc"
}
con=mysql.connector.connect(**config)
username="1 OR 1=1"
password="1 OR 1=1"
sql="SELECT COUNT(*) FROM t_user WHERE username=%s " \
"AND AES_DECRYPT(UNHEX(password),'HelloWorld')=%s";
cursor=con.cursor()
cursor.execute(sql,(username,password))
print(cursor.fetchone()[0])
con.close()
data:image/s3,"s3://crabby-images/a3e34/a3e34f05302a5958adbe123b60cc4562cb3729d3" alt="在这里插入图片描述"
异常处理
con.start_transaction([事物隔离级别]) con.commit() 事物提交 con.rollback() 事物回滚 未写入数据前:
mysql> select * from t_emp;
+
| empno | ename | job | mgr | hiredate | sal | comm | deptno |
+
| 7369 | SMITH | CLERK | 7902 | 1980-12-17 | 800.00 | NULL | 20 |
| 7499 | ALLEN | SALESMAN | 7698 | 1981-02-20 | 1600.00 | 300.00 | 30 |
| 7521 | WARD | SALESMAN | 7698 | 1981-02-22 | 1250.00 | 500.00 | 30 |
| 7566 | JONES | MANAGER | 7839 | 1981-04-02 | 2975.00 | NULL | 20 |
| 7654 | MARTIN | SALESMAN | 7698 | 1981-09-28 | 1250.00 | 1400.00 | 30 |
| 7698 | BLAKE | MANAGER | 7839 | 1981-05-01 | 2850.00 | NULL | 30 |
| 7782 | CLARK | MANAGER | 7839 | 1981-06-09 | 2450.00 | NULL | 10 |
| 7788 | SCOTT | ANALYST | 7566 | 1982-12-09 | 3000.00 | NULL | 20 |
| 7839 | KING | PRESIDENT | NULL | 1981-11-17 | 5000.00 | NULL | 10 |
| 7844 | TURNER | SALESMAN | 7698 | 1981-09-08 | 1500.00 | 0.00 | 30 |
| 7876 | ADAMS | CLERK | 7788 | 1983-01-12 | 1100.00 | NULL | 20 |
| 7900 | JAMES | CLERK | 7698 | 1981-12-03 | 950.00 | NULL | 30 |
| 7902 | FORD | ANALYST | 7566 | 1981-12-03 | 3000.00 | NULL | 20 |
| 7934 | MILLER | CLERK | 7782 | 1982-01-23 | 1300.00 | NULL | 10 |
+
14 rows in set (0.00 sec)
写入数据
import mysql.connector
try:
con=mysql.connector.connect(
host="localhost",
port=3306,
user="root",
password="cheng.12345",
database="gpc"
)
con.start_transaction()
cursor=con.cursor()
sql="INSERT INTO t_emp(empno,ename,job,mgr,hiredate,sal,comm,deptno) " \
"VALUES(%s,%s,%s,%s,%s,%s,%s,%s)"
cursor.execute(sql,(9600,"赵娜","SALESMAN",None,"1985-12-1",2500,None,10))
con.commit()
except Exception as e:
if "con" in dir():
con.rollback()
print(e)
finally:
if "con" in dir():
con.close()
data:image/s3,"s3://crabby-images/c5700/c5700f851ec67f028078a9762b609c7233d6237b" alt="在这里插入图片描述"
mysql> select * from t_emp;
+
| empno | ename | job | mgr | hiredate | sal | comm | deptno |
+
| 7369 | SMITH | CLERK | 7902 | 1980-12-17 | 800.00 | NULL | 20 |
| 7499 | ALLEN | SALESMAN | 7698 | 1981-02-20 | 1600.00 | 300.00 | 30 |
| 7521 | WARD | SALESMAN | 7698 | 1981-02-22 | 1250.00 | 500.00 | 30 |
| 7566 | JONES | MANAGER | 7839 | 1981-04-02 | 2975.00 | NULL | 20 |
| 7654 | MARTIN | SALESMAN | 7698 | 1981-09-28 | 1250.00 | 1400.00 | 30 |
| 7698 | BLAKE | MANAGER | 7839 | 1981-05-01 | 2850.00 | NULL | 30 |
| 7782 | CLARK | MANAGER | 7839 | 1981-06-09 | 2450.00 | NULL | 10 |
| 7788 | SCOTT | ANALYST | 7566 | 1982-12-09 | 3000.00 | NULL | 20 |
| 7839 | KING | PRESIDENT | NULL | 1981-11-17 | 5000.00 | NULL | 10 |
| 7844 | TURNER | SALESMAN | 7698 | 1981-09-08 | 1500.00 | 0.00 | 30 |
| 7876 | ADAMS | CLERK | 7788 | 1983-01-12 | 1100.00 | NULL | 20 |
| 7900 | JAMES | CLERK | 7698 | 1981-12-03 | 950.00 | NULL | 30 |
| 7902 | FORD | ANALYST | 7566 | 1981-12-03 | 3000.00 | NULL | 20 |
| 7934 | MILLER | CLERK | 7782 | 1982-01-23 | 1300.00 | NULL | 10 |
| 9600 | 赵娜 | SALESMAN | NULL | 1985-12-01 | 2500.00 | NULL | 10 |
+
15 rows in set (0.00 sec)
数据库连接池
- 数据库连接是一种关键的,有限的,昂贵的资源,在并发执行的应用程序中体现得尤为突出
- tcp连接需要三次握手,四次挥手,然后数据库还要验证用户信息
- 数据库连接池,预先创建出一些数据库连接,然后缓存起来,避免了程序语言反复创建和销毁连接昂贵代价
data:image/s3,"s3://crabby-images/93974/9397419e92b34f7975f184628a02bcaa070c9147" alt="在这里插入图片描述"
data:image/s3,"s3://crabby-images/cbfba/cbfba5a92420acb6912d745153316e80049db52e" alt="在这里插入图片描述"
import mysql.connector.pooling
config={
"host":"localhost",
"port":3306,
"user":"root",
"password":"cheng.123455",
"database":"gpc"
}
try:
pool=mysql.connector.pooling.MySQLConnectionPool(
**config,
pool_size=10
)
con=pool.get_connection()
con.start_transaction()
cursor=con.cursor()
sql="update t_emp set sal=sal+%s where deptno=%s"
cursor.execute(sql,(200000,20))
con.commit()
except Exception as e:
if "con" in dir():
con.rollback()
print(e)
data:image/s3,"s3://crabby-images/702c8/702c84dcf3d32deadf0ff61986b5fa299ca57ab2" alt="在这里插入图片描述"
删除数据
data:image/s3,"s3://crabby-images/2a102/2a1025352ceb53edf7aeb2ecaa83e96242da73cd" alt="在这里插入图片描述"
import mysql.connector.pooling
config={
"host":"localhost",
"port":3306,
"user":"root",
"password":"cheng.1333",
"database":"gpc"
}
try:
pool=mysql.connector.pooling.MySQLConnectionPool(
**config,
pool_size=10
)
con=pool.get_connection()
con.start_transaction()
cursor=con.cursor()
sql="delete from t_emp where deptno = 20; "
cursor.execute(sql)
con.commit()
except Exception as e:
if "con" in dir():
con.rollback()
print(e)
data:image/s3,"s3://crabby-images/889d7/889d7fd3cb8b65a5ffb03009ac73e2cf7bfae104" alt="在这里插入图片描述" 清空
import mysql.connector.pooling
config={
"host":"localhost",
"port":3306,
"user":"root",
"password":"cheng.12333",
"database":"gpc"
}
try:
pool=mysql.connector.pooling.MySQLConnectionPool(
**config,
pool_size=10
)
con=pool.get_connection()
con.start_transaction()
cursor=con.cursor()
sql="TRUNCATE TABLE t_emp;"
cursor.execute(sql)
con.commit()
except Exception as e:
if "con" in dir():
con.rollback()
print(e)
data:image/s3,"s3://crabby-images/cdcd1/cdcd1c5c468717dafbddf4895221bd7863b97b5e" alt="在这里插入图片描述"
循环执行sql语句
executemany()
import mysql.connector.pooling
config={
"host":"localhost",
"port":3306,
"user":"root",
"password":"cheng.1023",
"database":"gpc"
}
try:
pool=mysql.connector.pooling.MySQLConnectionPool(
**config,
pool_size=10
)
con=pool.get_connection()
con.start_transaction()
cursor=con.cursor()
sql="INSERT INTO t_dept(deptno,dname,loc) VALUES(%s,%s,%s)"
data=[
[100,"A部门","北京"],[110,"B部门","上海"]
]
cursor.executemany(sql,data)
con.commit()
except Exception as e:
if "con" in dir():
con.rollback()
print(e)
data:image/s3,"s3://crabby-images/d5668/d566845db8bebe250fbdad3c2a964877310bca73" alt="在这里插入图片描述"
案例
使用insert 语句,把部门平均底薪超过公司平均底薪的这样部门里的员工信息导入到t_emp_new表里面,并且让这些员工隶属于sales部门
一阶段
原: data:image/s3,"s3://crabby-images/17676/17676bacb988b13ff24cf9d4a249b2ea82db5f56" alt="在这里插入图片描述"
import mysql.connector.pooling
config={
"host":"localhost",
"port":3306,
"user":"root",
"password":"cheng",
"database":"gpc"
}
try:
pool = mysql.connector.pooling.MySQLConnectionPool(
**config,
pool_size=10
)
con = pool.get_connection()
con.start_transaction()
cursor = con.cursor()
sql = "CREATE TABLE t_emp_new AS (SELECT * FROM t_emp)"
cursor.execute(sql)
con.commit()
except Exception as e:
print(e)
现: data:image/s3,"s3://crabby-images/fb962/fb962ef026ec1774f1d44137b652265312c7c688" alt="在这里插入图片描述"
二阶段
import mysql.connector.pooling
config={
"host":"localhost",
"port":3306,
"user":"root",
"password":"cheng",
"database":"gpc"
}
try:
pool = mysql.connector.pooling.MySQLConnectionPool(
**config,
pool_size=10
)
con = pool.get_connection()
con.start_transaction()
cursor = con.cursor()
sql = "DROP TABLE IF EXISTS t_emp_new"
cursor.execute(sql)
sql = "CREATE TABLE t_emp_new LIKE t_emp"
cursor.execute(sql)
sql="SELECT AVG(sal) AS avg FROM t_emp"
cursor.execute(sql)
temp = cursor.fetchone()
avg = temp[0]
print("公司平均底薪:",temp)
sql="SELECT deptno FROM t_emp GROUP BY deptno HAVING AVG(sal)>=%s"
cursor.execute(sql,[avg])
for one in cursor:
print(one)
con.commit()
except Exception as e:
print(e)
data:image/s3,"s3://crabby-images/12079/12079d8f4d8b0ddaea235bde26a5c6263a1334bf" alt="在这里插入图片描述" 插入
import mysql.connector.pooling
config={
"host":"localhost",
"port":3306,
"user":"root",
"password":"cheng.1023",
"database":"gpc"
}
try:
pool = mysql.connector.pooling.MySQLConnectionPool(
**config,
pool_size=10
)
con = pool.get_connection()
con.start_transaction()
cursor = con.cursor()
sql = "DROP TABLE IF EXISTS t_emp_new"
cursor.execute(sql)
sql = "CREATE TABLE t_emp_new LIKE t_emp"
cursor.execute(sql)
sql="SELECT AVG(sal) AS avg FROM t_emp"
cursor.execute(sql)
temp = cursor.fetchone()
avg = temp[0]
print("公司平均底薪:",temp)
sql="SELECT deptno FROM t_emp GROUP BY deptno HAVING AVG(sal)>=%s"
cursor.execute(sql,[avg])
temp = cursor.fetchall()
print(temp)
sql="INSERT INTO t_emp_new SELECT * FROM t_emp WHERE deptno IN ("
for index in range(len(temp)):
one = temp[index][0]
if index < len(temp)-1:
sql+=str(one)+","
else:
sql+=str(one)
sql+=")"
print(sql)
cursor.execute(sql)
con.commit()
except Exception as e:
print(e)
data:image/s3,"s3://crabby-images/d2eb0/d2eb05453df7ef2b54bd21864cfca8b47cd2e43c" alt="在这里插入图片描述"
三阶段
data:image/s3,"s3://crabby-images/3b07d/3b07dccbca1648d4d1c647104a827ee47d0d3fba" alt="在这里插入图片描述"
import mysql.connector.pooling
config={
"host":"localhost",
"port":3306,
"user":"root",
"password":"cheng",
"database":"gpc"
}
try:
pool = mysql.connector.pooling.MySQLConnectionPool(
**config,
pool_size=10
)
con = pool.get_connection()
con.start_transaction()
cursor = con.cursor()
sql = "DROP TABLE IF EXISTS t_emp_new"
cursor.execute(sql)
sql = "CREATE TABLE t_emp_new LIKE t_emp"
cursor.execute(sql)
sql="SELECT AVG(sal) AS avg FROM t_emp"
cursor.execute(sql)
temp = cursor.fetchone()
avg = temp[0]
print("公司平均底薪:",temp)
sql="SELECT deptno FROM t_emp GROUP BY deptno HAVING AVG(sal)>=%s"
cursor.execute(sql,[avg])
temp = cursor.fetchall()
print(temp)
sql="INSERT INTO t_emp_new SELECT * FROM t_emp WHERE deptno IN ("
for index in range(len(temp)):
one = temp[index][0]
if index < len(temp)-1:
sql+=str(one)+","
else:
sql+=str(one)
sql+=")"
print(sql)
cursor.execute(sql)
sql="DELETE FROM t_emp WHERE deptno IN ("
for index in range(len(temp)):
one=temp[index][0]
if index < len(temp)-1:
sql+=str(one)+","
else:
sql+=str(one)
sql+=")"
print(sql)
cursor.execute(sql)
con.commit()
except Exception as e:
if "con" in dir():
con.rollback()
print(e)
data:image/s3,"s3://crabby-images/1a830/1a830bed3be4385684e436cae1c3945c49544c20" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/ac7d9/ac7d988a99daf409474b7b9b9868e14a29a974dc" alt="在这里插入图片描述"
最终
import mysql.connector.pooling
config={
"host":"localhost",
"port":3306,
"user":"root",
"password":"cheng",
"database":"gpc"
}
try:
pool = mysql.connector.pooling.MySQLConnectionPool(
**config,
pool_size=10
)
con = pool.get_connection()
con.start_transaction()
cursor = con.cursor()
sql = "DROP TABLE IF EXISTS t_emp_new"
cursor.execute(sql)
sql = "CREATE TABLE t_emp_new LIKE t_emp"
cursor.execute(sql)
sql="SELECT AVG(sal) AS avg FROM t_emp"
cursor.execute(sql)
temp = cursor.fetchone()
avg = temp[0]
print("公司平均底薪:",temp)
sql="SELECT deptno FROM t_emp GROUP BY deptno HAVING AVG(sal)>=%s"
cursor.execute(sql,[avg])
temp = cursor.fetchall()
print(temp)
sql="INSERT INTO t_emp_new SELECT * FROM t_emp WHERE deptno IN ("
for index in range(len(temp)):
one = temp[index][0]
if index < len(temp)-1:
sql+=str(one)+","
else:
sql+=str(one)
sql+=")"
print(sql)
cursor.execute(sql)
sql="DELETE FROM t_emp WHERE deptno IN ("
for index in range(len(temp)):
one=temp[index][0]
if index < len(temp)-1:
sql+=str(one)+","
else:
sql+=str(one)
sql+=")"
print(sql)
cursor.execute(sql)
sql="SELECT deptno FROM t_dept WHERE dname=%s"
cursor.execute(sql,["SALES"])
deptno=cursor.fetchone()[0]
sql="UPDATE t_emp_new SET deptno=%s"
cursor.execute(sql,[deptno])
con.commit()
except Exception as e:
if "con" in dir():
con.rollback()
print(e)
data:image/s3,"s3://crabby-images/a4b4e/a4b4e25b7c3ea0525208a1b027f99a48fa89930a" alt="在这里插入图片描述"
data:image/s3,"s3://crabby-images/228c4/228c4b09171234087772472d864352475589088b" alt="在这里插入图片描述"
|