python mysql操作_Python的MySQL操作
Python的DB-API,為大多數(shù)的數(shù)據(jù)庫實現(xiàn)了接口,使用它連接各數(shù)據(jù)庫后,就可以用相同的方式操作各數(shù)據(jù)庫。
Python DB-API使用流程:
引入API模塊。
獲取與數(shù)據(jù)庫的連接。
執(zhí)行SQL語句和存儲過程。
關(guān)閉數(shù)據(jù)庫連接。
一、安裝MySQL客戶端
MySQLdb 是用于Python鏈接Mysql數(shù)據(jù)庫的接口,它實現(xiàn)了 Python 數(shù)據(jù)庫 API 規(guī)范 V2.0,基于 MySQL C API 上建立的。
如果是windows系統(tǒng):登錄https://pypi.python.org/pypi/MySQL-python/1.2.5找到.exe結(jié)尾的包,下載安裝就好了,然后在cmd中執(zhí)行:
如果結(jié)果如上圖所示,則說明安裝成功了。
如果是Linux系統(tǒng),可以下載源碼包進行安裝,https://pypi.python.org/pypi/MySQL-python/1.2.5中下載zip包,然后安裝:
yum install –y python-devel
yum install –y mysql-devel
yum install –y gcc
unzip MySQL-python-1.2.5.zip
cd MySQL-python-1.2.5python setup.py build
python setup.py install
python>>> import MySQLdb
二、數(shù)據(jù)庫連接
MySQLdb提供了connect方法用來和數(shù)據(jù)庫建立連接,接收數(shù)個參數(shù),返回連接對象:代碼如下:
首先在mysql的數(shù)據(jù)庫中建立python庫
create database python;
conn=MySQLdb.connect(host="192.168.203.12",user="momo",passwd="123456",db="python",charset="utf8")
比較常用的參數(shù)包括:
host:數(shù)據(jù)庫主機名.默認是用本地主機
user:數(shù)據(jù)庫登陸名.默認是當前用戶
passwd:數(shù)據(jù)庫登陸的秘密.默認為空
db:要使用的數(shù)據(jù)庫名.沒有默認值
port:MySQL服務(wù)使用的TCP端口.默認是3306,數(shù)字類型
charset:數(shù)據(jù)庫編碼
推薦把所有數(shù)據(jù)庫的配置寫在一個字典中,如下所示:
defconnect_mysql():
db_config={'host': '192.168.203.12','port': 3306,'user': 'momo','passwd': '123456','db': 'python','charset': 'utf8'}
cnx= MySQLdb.connect(**db_config)return cnx
三、MySQL事物
MySQL 事務(wù)主要用于處理操作量大,復(fù)雜度高的數(shù)據(jù)。比如,你操作一個數(shù)據(jù)庫,公司的一個員工離職了,你要在數(shù)據(jù)庫中刪除他的資料,也要刪除該人員相關(guān)的,比如郵箱,個人資產(chǎn)等。這些數(shù)據(jù)庫操作語言就構(gòu)成了一個事務(wù)。
在MySQL中只有使用了Innodb數(shù)據(jù)庫引擎的數(shù)據(jù)庫或表才支持事務(wù),所以很多情況下我們都使用innodb引擎。
事務(wù)處理可以用來維護數(shù)據(jù)庫的完整性,保證成批的SQL語句要么全部執(zhí)行,要么全部不執(zhí)行。
一般來說,事務(wù)是必須滿足4個條件(ACID): Atomicity(原子性)、Consistency(穩(wěn)定性)、Isolation(隔離性)、Durability(可靠性)
1、事務(wù)的原子性:一組事務(wù),要么成功;要么撤回。
2、穩(wěn)定性 : 有非法數(shù)據(jù)(外鍵約束之類),事務(wù)撤回。
3、隔離性:事務(wù)獨立運行。一個事務(wù)處理后的結(jié)果,影響了其他事務(wù),那么其他事務(wù)會撤回。事務(wù)的100%隔離,需要犧牲速度。
4、可靠性:軟、硬件崩潰后,InnoDB數(shù)據(jù)表驅(qū)動會利用日志文件重構(gòu)修改。可靠性和高速度不可兼得, innodb_flush_log_at_trx_commit選項 決定什么時候吧事務(wù)保存到日志里。
而mysql在默認的情況下,他是把每個select,insert,update,delete等做為一個事務(wù)的,登錄mysql服務(wù)器,進入mysql,執(zhí)行以下命令:
mysql> show variables like 'auto%';+--------------------------+-------+
| Variable_name | Value |
+--------------------------+-------+
| auto_increment_increment | 1 |
| auto_increment_offset | 1 |
| autocommit | ON |
| automatic_sp_privileges | ON |
+--------------------------+-------+
4 rows in set (0.00 sec)
如上所示: 有一個參數(shù)autocommit就是自動提交的意思,每執(zhí)行一個msyql的select,insert,update等操作,就會進行自動提交。
如果把改選項關(guān)閉,我們就可以每次執(zhí)行完一次代碼就需要進行手動提交,connect對象給我們提供了兩種辦法來操作提交數(shù)據(jù)。
a)??????? mysql事務(wù)的方法
commit():提交當前事務(wù),如果是支持事務(wù)的數(shù)據(jù)庫執(zhí)行增刪改后沒有commit則數(shù)據(jù)庫默認回滾,白操作了
rollback():取消當前事務(wù)
下面我們來看個例子:
我們先創(chuàng)建一個員工表:
create table employees (
emp_no intnotnull auto_increment,
emp_name varchar(16) notnull,
gender enum('M', 'F') notnull,
hire_date datenotnull,
primary key (emp_no)
);
其中,emp_no為員工id,為主鍵且唯一
emp_name為:員工的名字
fender為:性別,只有M和F兩種選擇
hire_date為:雇傭的時間。
為了試驗的效果,我們插入幾條數(shù)據(jù):
insert into employees(emp_no, emp_name, gender, hire_date) values(1001, 'lingjiang', 'M', '2015-04-01');
insert into employees(emp_no, emp_name, gender, hire_date) values(1002, 'xiang', 'M', '2015-04-01');
insert into employees(emp_no, emp_name, gender, hire_date) values(1003, 'shang', 'M', '2015-04-01');
mysql> select * fromemployees;+--------+-----------+--------+------------+
| emp_no | emp_name | gender | hire_date |
+--------+-----------+--------+------------+
| 1001 | lingjiang | M | 2015-04-01 |
| 1002 | xiang | M | 2015-04-01 |
| 1003 | shang | M | 2015-04-01 |
+--------+-----------+--------+------------+e) rowsin set (0.00 sec)
四、MySQL游標
游標(cursor)
游標是系統(tǒng)為用戶開設(shè)的一個數(shù)據(jù)緩沖區(qū),存放SQL語句的執(zhí)行結(jié)果,用戶可以用SQL語句逐一從游標中獲取記錄,并賦給主變量,交由python進一步處理,一組主變量一次只能存放一條記錄,僅使用主變量并不能完全滿足SQL語句向應(yīng)用程序輸出數(shù)據(jù)的要求。
游標和游標的優(yōu)點:在數(shù)據(jù)庫中,游標是一個十分重要的概念。游標提供了一種對從表中檢索出的數(shù)據(jù)進行操作的靈活手段,就本質(zhì)而言,游標實際上是一種能從包括多條數(shù)據(jù)記錄的結(jié)果集中每次提取一條記錄的機制。游標總是與一條SQL 選擇語句相關(guān)聯(lián)因為游標由結(jié)果集(可以是零條、一條或由相關(guān)的選擇語句檢索出的多條記錄)和結(jié)果集中指向特定記錄的游標位置組成。當決定對結(jié)果集進行處理時,必須聲明一個指向該結(jié)果集的游標。
常用方法:
cursor():創(chuàng)建游標對象
close():關(guān)閉此游標對象
fetchone():得到結(jié)果集的下一行
fetchmany([size = cursor.arraysize]):得到結(jié)果集的下幾行
fetchall():得到結(jié)果集中剩下的所有行
excute(sql[, args]):執(zhí)行一個數(shù)據(jù)庫查詢或命令
executemany (sql, args):執(zhí)行多個數(shù)據(jù)庫查詢或命令
程序例子:
1.創(chuàng)建游標對象
import MySQLdb
db_config={'host': '192.168.203.12','port': 3306,'user': 'momo','passwd': '123456','db': 'python','charset': 'utf8'}
cnx= MySQLdb.connect(**db_config)
cus= cnx.cursor()
2.對游標的基本操作
importMySQLdbdefconnect_mysql():
db_config={'host': '192.168.203.12','port': 3306,'user': 'momo','passwd': '123456','db': 'python','charset': 'utf8'}
cnx= MySQLdb.connect(**db_config)returncnxif __name__ == '__main__':
cnx=connect_mysql()
cus=cnx.cursor()
sql= '''select * from employees;'''
try:
cus.execute(sql)
result1=cus.fetchone()print('result1:')print(result1)
result2= cus.fetchmany(1)print('result2:')print(result2)
result3=cus.fetchall()print('result3:')print(result3) cus.close()
cnx.commit()exceptException as e:
cnx.rollback()print('error')raiseefinally:
cnx.close()
結(jié)果:
result1:
(1001L, u'lingjiang', u'M', datetime.date(2015, 4, 1))
result2:
((1002L, u'xiang', u'M', datetime.date(2015, 4, 1)),)
result3:
((1003L, u'shang', u'M', datetime.date(2015, 4, 1)),)
解釋:
先通過MySQLdb.connect(**db_config)建立mysql連接對象
在通過 = cnx.cursor()創(chuàng)建游標
fetchone():在最終搜索的數(shù)據(jù)中去一條數(shù)據(jù)
fetchmany(1)在接下來的數(shù)據(jù)中在去1行的數(shù)據(jù),這個數(shù)字可以自定義,定義多少就是在結(jié)果集中取多少條數(shù)據(jù)。
fetchall()是在所有的結(jié)果中搞出來所有的數(shù)據(jù)。
3.執(zhí)行多行SQL語句
#!/usr/bin/env python#-*- coding:utf-8 -*-#@Time : 2017/9/18 22:17#@Author : lingxiangxiang#@File : domon3.py
from demon2 importconnect_mysqlimportMySQLdbdefconnect_mysql():
db_config={"host": "192.168.203.12","port": 3306,"user": "momo","passwd": "123456","db": "python","charset": "utf8"}try:
cnx= MySQLdb.connect(**db_config)exceptException as e:raiseereturncnxif __name__ == "__main__":
sql= "select * from tmp;"sql1= "insert into tmp(id) value (%s);"param=[]for i in xrange(100, 130):
param.append([str(i)])print(param)
cnx=connect_mysql()
cus=cnx.cursor()print(dir(cus))try:
cus.execute(sql)
cus.executemany(sql1, param)#help(cus.executemany)
result1 =cus.fetchone()print("result1")print(result1)
result2= cus.fetchmany(3)print("result2")print(result2)
result3=cus.fetchall()print("result3")print(result3)
cus.close()
cnx.commit()exceptException as e:
cnx.rollback()raiseefinally:
cnx.close()
五、數(shù)據(jù)庫連接池
python編程中可以使用MySQLdb進行數(shù)據(jù)庫的連接及諸如查詢/插入/更新等操作,但是每次連接mysql數(shù)據(jù)庫請求時,都是獨立的去請求訪問,相當浪費資源,而且
訪問數(shù)量達到一定數(shù)量時,對mysql的性能會產(chǎn)生較大的影響。因此,實際使用中,通常會使用數(shù)據(jù)庫的連接池技術(shù),來訪問數(shù)據(jù)庫達到資源復(fù)用的目的。
python的數(shù)據(jù)庫連接池包 DBUtils:
DBUtils是一套Python數(shù)據(jù)庫連接池包,并允許對非線程安全的數(shù)據(jù)庫接口進行線程安全包裝。DBUtils來自Webware for Python。
DBUtils提供兩種外部接口:
* PersistentDB :提供線程專用的數(shù)據(jù)庫連接,并自動管理連接。
* PooledDB :提供線程間可共享的數(shù)據(jù)庫連接,并自動管理連接。
PooledDB的參數(shù):
1. mincached,最少的空閑連接數(shù),如果空閑連接數(shù)小于這個數(shù),pool會創(chuàng)建一個新的連接
2. maxcached,最大的空閑連接數(shù),如果空閑連接數(shù)大于這個數(shù),pool會關(guān)閉空閑連接
3. maxconnections,最大的連接數(shù),
4. blocking,當連接數(shù)達到最大的連接數(shù)時,在請求連接的時候,如果這個值是True,
請求連接的程序會一直等待,直到當前連接數(shù)小于最大連接數(shù),如果這個值是False,會報錯,
5. maxshared 當連接數(shù)達到這個數(shù),新請求的連接會分享已經(jīng)分配出去的連接
在uwsgi中,每個http請求都會分發(fā)給一個進程,連接池中配置的連接數(shù)都是一個進程為單位的(即上面的最大連接數(shù),都是在一個進程中的連接數(shù)),
而如果業(yè)務(wù)中,一個http請求中需要的sql連接數(shù)不是很多的話(其實大多數(shù)都只需要創(chuàng)建一個連接),配置的連接數(shù)配置都不需要太大。
連接池對性能的提升表現(xiàn)在:
1.在程序創(chuàng)建連接的時候,可以從一個空閑的連接中獲取,不需要重新初始化連接,提升獲取連接的速度
2.關(guān)閉連接的時候,把連接放回連接池,而不是真正的關(guān)閉,所以可以減少頻繁地打開和關(guān)閉連接
六、數(shù)據(jù)庫的操作
1.建表
各個表的結(jié)構(gòu)如下:
student表:
字段名
類型
是否為空
主鍵
描述
StdID
int
否
是
學(xué)生ID
StdName
varchar(100)
否
學(xué)生姓名
Gender
enum('M', 'F')
是
性別
Age
tinyint
是
年齡
course表:
字段名
類型
是否為空
主鍵
描述
CouID
int
否
是
課程ID
Cname
varchar(50)
否
課程名字
TID
int
否
老師ID
Score表:
字段名
類型
是否為空
主鍵
描述
SID
int
否
是
分數(shù)ID
StdID
int
否
學(xué)生id
CouID
int
否
課程id
Grade
int
否
分數(shù)
teacher表:
字段名
類型
是否為空
主鍵
描述
TID
int
否
是
老師ID
Tname
varcher(100)
否
老師名字
在Linux中MySQL建立student表,然后在Python代碼中執(zhí)行:
importMySQLdbdefconnect_mysql():
db_config={'host': '192.168.203.12','port': 3306,'user': 'momo','passwd': '123456','db': 'student','charset': 'utf8'}
cnx= MySQLdb.connect(**db_config)returncnxif __name__ == '__main__':
cnx=connect_mysql()
cus=cnx.cursor()#sql = '''insert into student(id, name, age, gender, score) values ('1001', 'ling', 29, 'M', 88), ('1002', 'ajing', 29, 'M', 90), ('1003', 'xiang', 33, 'M', 87);'''
student = '''create table Student(
StdID int not null,
StdName varchar(100) not null,
Gender enum('M', 'F'),
Age tinyint
)'''course= '''create table Course(
CouID int not null,
CName varchar(50) not null,
TID int not null
)'''score= '''create table Score(
SID int not null,
StdID int not null,
CID int not null,
Grade int not null
)'''teacher= '''create table Teacher(
TID int not null,
TName varchar(100) not null
)'''tmp= '''set @i := 0;
create table tmp as select (@i := @i + 1) as id from information_schema.tables limit 10;'''
try:
cus.execute(student)
cus.execute(course)
cus.execute(score)
cus.execute(thearch)
cus.execute(tmp)
cus.close()
cnx.commit()exceptException as e:
cnx.rollback()print('error')raiseefinally:
cnx.close()
結(jié)果:
mysql> show tables;
+------------------+
| Tables_in_python |
+------------------+
| Course?????????? |
| Score??????????? |
| Student????????? |
| Teacher????????? |
| tmp????????????? |
+------------------+
1??????? rows in set (0.00 sec)
information_schema數(shù)據(jù)庫表說明:
SCHEMATA表:提供了當前mysql實例中所有數(shù)據(jù)庫的信息。是show databases的結(jié)果取之此表。
TABLES表:提供了關(guān)于數(shù)據(jù)庫中的表的信息(包括視圖)。詳細表述了某個表屬于哪個schema,表類型,表引擎,創(chuàng)建時間等信息。是show tables from schemaname的結(jié)果取之此表。
COLUMNS表:提供了表中的列信息。詳細表述了某張表的所有列以及每個列的信息。是show columns from schemaname.tablename的結(jié)果取之此表。
STATISTICS表:提供了關(guān)于表索引的信息。是show index from schemaname.tablename的結(jié)果取之此表。
USER_PRIVILEGES(用戶權(quán)限)表:給出了關(guān)于全程權(quán)限的信息。該信息源自mysql.user授權(quán)表。是非標準表。
SCHEMA_PRIVILEGES(方案權(quán)限)表:給出了關(guān)于方案(數(shù)據(jù)庫)權(quán)限的信息。該信息來自mysql.db授權(quán)表。是非標準表。
TABLE_PRIVILEGES(表權(quán)限)表:給出了關(guān)于表權(quán)限的信息。該信息源自mysql.tables_priv授權(quán)表。是非標準表。
COLUMN_PRIVILEGES(列權(quán)限)表:給出了關(guān)于列權(quán)限的信息。該信息源自mysql.columns_priv授權(quán)表。是非標準表。
CHARACTER_SETS(字符集)表:提供了mysql實例可用字符集的信息。是SHOW CHARACTER SET結(jié)果集取之此表。
COLLATIONS表:提供了關(guān)于各字符集的對照信息。
COLLATION_CHARACTER_SET_APPLICABILITY表:指明了可用于校對的字符集。這些列等效于SHOW COLLATION的前兩個顯示字段。
TABLE_CONSTRAINTS表:描述了存在約束的表。以及表的約束類型。
KEY_COLUMN_USAGE表:描述了具有約束的鍵列。
ROUTINES表:提供了關(guān)于存儲子程序(存儲程序和函數(shù))的信息。此時,ROUTINES表不包含自定義函數(shù)(UDF)。名為“mysql.proc name”的列指明了對應(yīng)于INFORMATION_SCHEMA.ROUTINES表的mysql.proc表列。
VIEWS表:給出了關(guān)于數(shù)據(jù)庫中的視圖的信息。需要有show views權(quán)限,否則無法查看視圖信息。
TRIGGERS表:提供了關(guān)于觸發(fā)程序的信息。必須有super權(quán)限才能查看該表
而TABLES在安裝好mysql的時候,一定是有數(shù)據(jù)的,因為在初始化mysql的時候,就需要創(chuàng)建系統(tǒng)表,該表一定有數(shù)據(jù)。
set?@i?:=?0;
create?table?tmp?as?select?(@i?:=?@i?+?1)?as?id?from?information_schema.tables?limit?10;
mysql中變量不用事前申明,在用的時候直接用“@變量名”使用就可以了。set這個是mysql中設(shè)置變量的特殊用法,當@i需要在select中使用的時候,必須加:,這樣就創(chuàng)建好了一個表tmp,查看tmp的數(shù)據(jù):
mysql> select * from tmp;
+------+
| id
|
+------+
|
1 |
|
2 |
|
3 |
|
4 |
|
5 |
|
6 |
|
7 |
|
8 |
|
9 |
|
10 |
+------+
10 rows in set (0.00 sec)
我們只是從information_schema.tables表中取10條數(shù)據(jù),任何表有10條數(shù)據(jù)也是可以的,然后把變量@i作為id列的值,分10次不斷輸出,依據(jù)最后select的結(jié)果,創(chuàng)建表tmp。
2.增加數(shù)據(jù)
substr是一個字符串函數(shù),從第二個參數(shù)1,開始取字符,取到3+ floor(rand() * 75)結(jié)束
floor函數(shù)代表的是去尾法取整數(shù)。
rand()函數(shù)代表的是從0到1取一個隨機的小數(shù)。
rand() * 75就代表的是:0到75任何一個小數(shù),
3+floor(rand() * 75)就代表的是:3到77的任意一個數(shù)字
concat()函數(shù)是一個對多個字符串拼接函數(shù)。
sha1是一個加密函數(shù),sha1(rand())對生成的0到1的一個隨機小數(shù)進行加密,轉(zhuǎn)換成字符串的形式。
concat(sha1(rand()), sha1(rand()))就代表的是:兩個0-1生成的小數(shù)加密然后進行拼接。
substr(concat(sha1(rand()), sha1(rand())), 1, floor(rand() * 80))就代表的是:從一個隨機生成的一個字符串的第一位開始取,取到(隨機3-77)位結(jié)束。
Gender字段:case floor(rand()*10) mod 2 when 1 then 'M' else 'F' end,就代表的是,
floor(rand()*10)代表0-9隨機取一個數(shù)
floor(rand()*10) mod 2 就是對0-9取得的隨機數(shù)除以2的余數(shù),
case floor(rand()*10) mod 2 when 1 then 'M' else 'F' end,代表:當余數(shù)為1是,就取M,其他的為F
Age字段:25-floor(rand() * 5)代表的就是,25減去一個0-4的一個整數(shù)
代碼如下:
importMySQLdbdefconnect_mysql():
db_config={'host': '192.168.203.12','port': 3306,'user': 'momo','passwd': '123456','db': 'student','charset': 'utf8'}
cnx= MySQLdb.connect(**db_config)returncnxif __name__ == '__main__':
cnx=connect_mysql()
students= '''set @i := 10000;
insert into Student select @i:=@i+1, substr(concat(sha1(rand()), sha1(rand())), 1, 3 + floor(rand() * 75)), case floor(rand()*10) mod 2 when 1 then 'M' else 'F' end, 25-floor(rand() * 5) from tmp a, tmp b, tmp c, tmp d;'''course= '''set @i := 10;
insert into Course select @i:=@i+1, substr(concat(sha1(rand()), sha1(rand())), 1, 5 + floor(rand() * 40)), 1 + floor(rand() * 100) from tmp a;'''score= '''set @i := 10000;
insert into Score select @i := @i +1, floor(10001 + rand()*10000), floor(11 + rand()*10), floor(1+rand()*100) from tmp a, tmp b, tmp c, tmp d;'''theacher= '''set @i := 100;
insert into Teacher select @i:=@i+1, substr(concat(sha1(rand()), sha1(rand())), 1, 5 + floor(rand() * 80)) from tmp a, tmp b;'''
try:
cus_students=cnx.cursor()
cus_students.execute(students)
cus_students.close()
cus_course=cnx.cursor()
cus_course.execute(course)
cus_course.close()
cus_score=cnx.cursor()
cus_score.execute(score)
cus_score.close()
cus_teacher=cnx.cursor()
cus_teacher.execute(theacher)
cus_teacher.close()
cnx.commit()exceptException as e:
cnx.rollback()print('error')raiseefinally:
cnx.close()
3.查數(shù)據(jù)
在數(shù)據(jù)庫中查出來所有名字有重復(fù)的同學(xué)的所有信息,然后寫入到文件中,代碼如下:
importcodecsimportMySQLdbdefconnect_mysql():
db_config={'host': '192.168.203.12','port': 3306,'user': 'momo','passwd': '123456','db': 'student','charset': 'utf8'}
cnx= MySQLdb.connect(**db_config)returncnxif __name__ == '__main__':
cnx=connect_mysql()
sql= '''select * from Student where StdName in (select StdName from Student group by StdName having count(1)>1 ) order by StdName;'''
try:
cus=cnx.cursor()
cus.execute(sql)
result=cus.fetchall()
with codecs.open('select.txt', 'w+') as f:for line inresult:
f.write(str(line))
f.write('\n')
cus.close()
cnx.commit()exceptException as e:
cnx.rollback()print('error')raiseefinally:
cnx.close()
解釋:
我們先來分析一下select查詢這個語句:select * from Student where StdName in (select StdName from Student group by StdName having count(1)>1 ) order by StdName;'
我們先來看括號里面的語句:select StdName from Student group by StdName having count(1)>1;這個是把所有學(xué)生名字重復(fù)的學(xué)生都列出來,
最外面select是套了一個子查詢,學(xué)生名字是在我們()里面的查出來的學(xué)生名字,把這些學(xué)生的所有信息都列出來。
result = cus.fetchall()列出結(jié)果以后,我們通過fetchall()函數(shù)把所有的內(nèi)容都取出來,這個result是一個tuple
通過文件寫入的方式,我們把取出來的result寫入到select.txt文件中。得到最終的結(jié)果
4.刪除數(shù)據(jù)
刪除課程成績最差的5名老師,刪除之前要先進行查詢。代碼如下:
importcodecsimportMySQLdbdefconnect_mysql():
db_config={'host': '192.168.203.12','port': 3306,'user': 'momo','passwd': '123456','db': 'student','charset': 'utf8'}
cnx= MySQLdb.connect(**db_config)returncnxif __name__ == '__main__':
cnx=connect_mysql()
sql= '''delete from Teacher where TID in(
select TID from (select Course.CouID, Course.TID, Teacher.TName, count(Teacher.TID) as count_teacher from Course
left join Score on Score.Grade < 60 and Course.CouID = Score.CouID
left join Teacher on Course.TID = Teacher.TID
group by Course.TID
order by count_teacher desc
limit 5) as test )'''
try:
cus=cnx.cursor()
cus.execute(sql)
result=cus.fetchall()
cus.close()
cnx.commit()exceptException as e:
cnx.rollback()print('error')raiseefinally:
cnx.close()
解釋:
先查詢出Course表中的Course.TID和Course.TID
left join 是關(guān)聯(lián)Score表,查出Score.Grade > 59,并且,課程ID和課程表的CouID要對應(yīng)上
left join Teacher 是關(guān)聯(lián)老師表,課程中的了老師ID和老師表中的老師ID對應(yīng)上
select中加上老師的名字Teacher.Tname和count(Teacher.TID)
group by Course.TID,在根據(jù)老師的的TID進行分組
oder by 最后對count_teacher進行排序,取前5行,
在通過套用一個select子查詢,把所有的TID摟出來
然后delete from Teacher 最后刪除TID在上表中的子查詢中
5.修改數(shù)據(jù)
把分數(shù)低于5分的成績加上60分,代碼如下:
importcodecsimportMySQLdbdefconnect_mysql():
db_config={'host': '192.168.203.12','port': 3306,'user': 'momo','passwd': '123456','db': 'student','charset': 'utf8'}
cnx= MySQLdb.connect(**db_config)returncnxif __name__ == '__main__':
cnx=connect_mysql()
sql= '''select *, (grade+60) as newGrade from Score where Grade <5;'''update= '''update Score set grade = grade + 60 where grade < 5;'''
try:
cus_start=cnx.cursor()
cus_start.execute(sql)
result1=cus_start.fetchall()print(len(result1))
cus_start.close()
cus_update=cnx.cursor()
cus_update.execute(update)
cus_update.close()
cus_end=cnx.cursor()
cus_end.execute(sql)
result2=cus_end.fetchall()print(len(result2))
cus_end.close()
cnx.commit()exceptException as e:
cnx.rollback()print('error')raiseefinally:
cnx.close()
解釋:
剛開始,我們可以查到分數(shù)小于5分的總個數(shù)有321個
select *, (grade+60) as newGrade from Score where Grade <5;這個sql是把所有的成績小于5的都列出來,然后最后加一列分數(shù)加60分的結(jié)果。
update Score set grade = grade + 60 where grade < 5;是把分數(shù)小于5的所有成績都加60分
最后在檢查分數(shù)小于5的個數(shù)為0,說明所有低于5分的分數(shù)都發(fā)生了改變。
6.索引
MySQL索引的概念
索引是一種特殊的文件(InnoDB數(shù)據(jù)表上的索引是表空間的一個組成部分),它們包含著對數(shù)據(jù)表里所有記錄的引用指針。更通俗的說,數(shù)據(jù)庫索引好比是一本書前面的目錄,能加快數(shù)據(jù)庫的查詢速度。
索引類別
普通索引
普通索引(由關(guān)鍵字 KEY 或 INDEX 定義的索引)的唯一任務(wù)是加快對數(shù)據(jù)的訪問速度。因此,應(yīng)該只為那些最經(jīng)常出現(xiàn)在查詢條件(WHERE column =)或排序條件(ORDER BY column)中的數(shù)據(jù)列創(chuàng)建索引。只要有可能,就應(yīng)該選擇一個數(shù)據(jù)最整齊、最緊湊的數(shù)據(jù)列(如一個整數(shù)類型的數(shù)據(jù)列)來創(chuàng)建索引。
唯一索引
普通索引允許被索引的數(shù)據(jù)列包含重復(fù)的值。比如說,因為人有可能同名,所以同一個姓名在同一個“員工個人資料”數(shù)據(jù)表里可能出現(xiàn)兩次或更多次。
如果能確定某個數(shù)據(jù)列將只包含彼此各不相同的值,在為這個數(shù)據(jù)列創(chuàng)建索引的時候就應(yīng)該用關(guān)鍵字UNIQUE 把它定義為一個唯一索引。這么做的好處:一是簡化了 MySQL 對這個索引的管理工作,這個索引也因此而變得更有效率;二是 MySQL 會在有新記錄插入數(shù)據(jù)表時,自動檢查新記錄的這個字段的值是否已經(jīng)在某個記錄的這個字段里出現(xiàn)過了;如果是,MySQL 將拒絕插入那條新記錄。也就是說,唯一索引可以保證數(shù)據(jù)記錄的唯一性。事實上,在許多場合,人們創(chuàng)建唯一索引的目的往往不是為了提高訪問速度,而只是為了避免數(shù)據(jù)出現(xiàn)重復(fù)。
主索引
在前面已經(jīng)反復(fù)多次強調(diào)過:必須為主鍵字段創(chuàng)建一個索引,這個索引就是所謂的“主索引”。主索引與唯一索引的唯一區(qū)別是:前者在定義時使用的關(guān)鍵字是 PRIMARY 而不是 UNIQUE。
外鍵索引
如果為某個外鍵字段定義了一個外鍵約束條件,MySQL 就會定義一個內(nèi)部索引來幫助自己以最有效率的方式去管理和使用外鍵約束條件。
復(fù)合索引
索引可以覆蓋多個數(shù)據(jù)列,如像 INDEX (columnA, columnB) 索引。這種索引的特點是 MySQL 可以有選擇地使用一個這樣的索引。如果查詢操作只需要用到 columnA 數(shù)據(jù)列上的一個索引,就可以使用復(fù)合索引 INDEX(columnA, columnB)。不過,這種用法僅適用于在復(fù)合索引中排列在前的數(shù)據(jù)列組合。比如說,INDEX (A,B,C) 可以當做 A 或 (A,B) 的索引來使用,但不能當做 B、C 或 (B,C) 的索引來使用。
mysql主鍵和索引的區(qū)別:
主鍵一定是唯一性索引,唯一性索引并不一定就是主鍵。
所謂主鍵就是能夠唯一標識表中某一行的屬性或?qū)傩越M,一個表只能有一個主鍵,但可以有多個候選索引。因為主鍵可以唯一標識某一行記錄,所以可以確保執(zhí)行數(shù)據(jù)更新、刪除的時候不會出現(xiàn)張冠李戴的錯誤。主鍵除了上述作用外,常常與外鍵構(gòu)成參照完整性約束,防止出現(xiàn)數(shù)據(jù)不一致。數(shù)據(jù)庫在設(shè)計時,主鍵起到了很重要的作用。主鍵可以保證記錄的唯一和主鍵域非空,數(shù)據(jù)庫管理系統(tǒng)對于主鍵自動生成唯一索引,所以主鍵也是一個特殊的索引。
一個表中可以有多個唯一性索引,但只能有一個主鍵。
主鍵列不允許空值,而唯一性索引列允許空值。
索引可以提高查詢的速度。
創(chuàng)建Course的CouID的字段為主鍵?? Score的SID字段為主鍵??? Student的StdID字段為主鍵? Teacher的TID字段為主鍵,代碼如下:
importcodecsimportMySQLdbdefconnect_mysql():
db_config={'host': '192.168.203.12','port': 3306,'user': 'momo','passwd': '123456','db': 'student','charset': 'utf8'}
cnx= MySQLdb.connect(**db_config)returncnxif __name__ == '__main__':
cnx=connect_mysql()
sql1= '''alter table Teacher add primary key(TID);'''sql2= '''alter table Student add primary key(StdID);'''sql3= '''alter table Score add primary key(SID);'''sql4= '''alter table Course add primary key(CouID);'''sql5= '''alter table Score add index idx_StdID_CouID(StdID, CouID);'''
#sql6 = '''alter table Score drop index idx_StdID_CouID;''' 刪除索引
sql7 = '''explain select * from Score where StdID = 16213;'''
try:
cus=cnx.cursor()
cus.execute(sql1)
cus.close()
cus=cnx.cursor()
cus.execute(sql2)
cus.close()
cus=cnx.cursor()
cus.execute(sql3)
cus.close()
cus=cnx.cursor()
cus.execute(sql4)
cus.close()
cus=cnx.cursor()
cus.execute(sql5)
cus.close()
cus=cnx.cursor()
cus.execute(sql7)
result=cus.fetchall()print(result)
cus.close()
cnx.commit()exceptException as e:
cnx.rollback()print('error')raiseefinally:
cnx.close()
結(jié)果:
((1L, u'SIMPLE', u'Score', u'ref', u'idx_StdID_CouID', u'idx_StdID_CouID', u'4', u'const', 4L, None),)
解釋:
Sql1, sql2, sql3, sql4是添加主鍵,sql5是增加一個索引,我們也可以在mysql的客戶端上執(zhí)行sq7,得到如下的結(jié)果:
mysql> explain select * from Score where StdID = 16213;
+----+-------------+-------+------+-----------------+-----------------+---------+-------+------+-------+
| id | select_type | table | type | possible_keys?? | key???????????? | key_len | ref?? | rows | Extra |
+----+-------------+-------+------+-----------------+-----------------+---------+-------+------+-------+
|? 1 | SIMPLE????? | Score | ref? | idx_StdID_CouID | idx_StdID_CouID | 4?????? | const |??? 4 | NULL? |
+----+-------------+-------+------+-----------------+-----------------+---------+-------+------+-------+
1 row in set (0.00 sec)
這個說明,我們在搜索StdID的時候,是走了idx_StdID_CouID索引的。
總結(jié)
以上是生活随笔為你收集整理的python mysql操作_Python的MySQL操作的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 男生qq网名帅气阳光133个
- 下一篇: pitr 原理_PostgreSQL热备