Python連接MySQL數(shù)據(jù)庫(kù)

Python標(biāo)準(zhǔn)數(shù)據(jù)庫(kù)接口為Python DB-API, Python DB-API為開(kāi)發(fā)人員提供了數(shù)據(jù)庫(kù)應(yīng)用 編程接口矾麻。

Python 數(shù)據(jù)庫(kù)接口支持非常多的數(shù)據(jù)庫(kù),你可以選擇適合你項(xiàng)目的數(shù)據(jù)庫(kù):

  • GadFly
  • mSQL
  • MySQL
  • PostgreSQL
  • Microsoft SQL Server 2000 Informix
  • Interbase Oracle Sybase

你可以訪問(wèn)Python數(shù)據(jù)庫(kù)接口及API查看詳細(xì)的支持?jǐn)?shù)據(jù)庫(kù)列表纱耻。

不同的數(shù)據(jù)庫(kù)你需要下載不同的DB API模塊,例如你需要訪問(wèn)Oracle數(shù)據(jù)庫(kù)和Mysql數(shù)據(jù),你需要下載Oracle和MySQL數(shù)據(jù)庫(kù)模塊。

DB-API是一個(gè)規(guī)范险耀。它定義了一系列必須的對(duì)象和數(shù)據(jù)庫(kù)存取方式, 以便為各種各樣的底層數(shù)據(jù)庫(kù)系統(tǒng)和多種多樣的數(shù)據(jù)庫(kù)接口程序提供一致的訪問(wèn)接口弄喘。

Python的DB-API,為大多數(shù)的數(shù)據(jù)庫(kù)實(shí)現(xiàn)了接口,使用它連接各數(shù)據(jù)庫(kù)后,就可以用相同 的方式操作各數(shù)據(jù)庫(kù)。

Python DB-API使用流程:

  1. 引入API模塊甩牺。
  2. 獲取與數(shù)據(jù)庫(kù)的連接蘑志。
  3. 執(zhí)行SQL語(yǔ)句和存儲(chǔ)過(guò)程。
  4. 關(guān)閉數(shù)據(jù)庫(kù)連接贬派。

1. MySQLdb

MySQLdb是用于Python鏈接Mysql數(shù)據(jù)庫(kù)的接口急但,它實(shí)現(xiàn)了Python 數(shù)據(jù)庫(kù)API規(guī)范V2.0,基于MySQL C API上建立的搞乏。

安裝

直接使用pip進(jìn)行安裝波桩,在此之前需要安裝一些系統(tǒng)依賴包。

  • CentOS
yum install gcc python-devel mysql-devel zlib-devel openssl-devel
  • Ubuntu
sudo apt-get install libmysqlclient-dev libmysqld-dev python-dev python-setuptools

安裝完依賴请敦,直接使用pip安裝镐躲,MySQLdb模塊的名字在pip上叫MySQL-python储玫。

pip install MySQL-python

常用函數(shù)

Python DB API 2.0 對(duì)事務(wù)提供了兩個(gè)方法:

  • commit() 提交
  • rollback() 回滾

cursor用來(lái)執(zhí)行命令的方法:

  • callproc(self, procname, args) 用來(lái)執(zhí)行存儲(chǔ)過(guò)程,接收的參數(shù)為存儲(chǔ)過(guò)程名和參數(shù)列表,返回值為受影響的行數(shù)
  • execute(self, query, args) 執(zhí)行單條sql語(yǔ)句,接收的參數(shù)為sql語(yǔ)句本身和使用的參數(shù)列表,返回值為受影響的行數(shù)
  • executemany(self, query, args) 執(zhí)行單挑sql語(yǔ)句,但是重復(fù)執(zhí)行參數(shù)列表里的參數(shù),返回值為受影響的行數(shù)
  • nextset(self) 移動(dòng)到下一個(gè)結(jié)果集

cursor用來(lái)接收返回值的方法:

  • fetchall(self) 接收全部的返回結(jié)果行.
  • fetchmany(self, size=None) 接收size條返回結(jié)果行.如果size的值大于返回的結(jié)果行的數(shù)量,則會(huì)返回cursor.arraysize條數(shù)據(jù).
  • fetchone(self) 返回一條結(jié)果行.
  • rowcount 這是一個(gè)只讀屬性,并返回執(zhí)行execute() 方法后影響的行數(shù)。
  • scroll(self, value, mode='relative') 移動(dòng)指針到某一行; 如果mode='relative',則表示從當(dāng)前所在行移動(dòng)value條,如果 mode='absolute',則表示從結(jié)果集的第一行移動(dòng)value條.

實(shí)例

#!/usr/bin/env python
# -*- coding: utf-8 -*-
import MySQLdb as mdb

# 連接數(shù)據(jù)庫(kù)
conn = mdb.connect('localhost', 'root', 'root')

# 也可以使用關(guān)鍵字參數(shù)
conn = mdb.connect(host='127.0.0.1', port=3306, user='root', passwd='root', db='test', charset='utf8')

# 也可以使用字典進(jìn)行連接參數(shù)的管理
config = {
    'host': '127.0.0.1',
    'port': 3306,
    'user': 'root',
    'passwd': 'root',
    'db': 'test',
    'charset': 'utf8'
}
conn = mdb.connect(**config)

# 如果使用事務(wù)引擎萤皂,可以設(shè)置自動(dòng)提交事務(wù)撒穷,或者在每次操作完成后手動(dòng)提交事務(wù)conn.commit()
conn.autocommit(1)    # conn.autocommit(True) 

# 使用cursor()方法獲取操作游標(biāo)
cursor = conn.cursor()
# 因該模塊底層其實(shí)是調(diào)用CAPI的,所以裆熙,需要先得到當(dāng)前指向數(shù)據(jù)庫(kù)的指針端礼。

try:
    # 創(chuàng)建數(shù)據(jù)庫(kù)
    DB_NAME = 'test'
    cursor.execute('DROP DATABASE IF EXISTS %s' %DB_NAME)
    cursor.execute('CREATE DATABASE IF NOT EXISTS %s' %DB_NAME)
    conn.select_db(DB_NAME)
    
    #創(chuàng)建表
    TABLE_NAME = 'user'
    cursor.execute('CREATE TABLE %s(id int primary key,name varchar(30))' %TABLE_NAME)
    
    # 插入單條數(shù)據(jù)
    sql = 'INSERT INTO user values("%d","%s")' %(1,"jack")
    
    # 不建議直接拼接sql,占位符方面可能會(huì)出問(wèn)題入录,execute提供了直接傳值
    value = [2,'John']
    cursor.execute('INSERT INTO test values(%s,%s)',value)
    
    # 批量插入數(shù)據(jù)
    values = []
    for i in range(3, 20):
        values.append((i,'kk'+str(i)))
    cursor.executemany('INSERT INTO user values(%s,%s)',values)
    
    # 查詢數(shù)據(jù)條目
    count = cursor.execute('SELECT * FROM %s' %TABLE_NAME)
    print 'total records: %d' %count
    print 'total records:', cursor.rowcount
    
    # 獲取表名信息
    desc = cursor.description
    print "%s %3s" % (desc[0][0], desc[1][0])
    
    # 查詢一條記錄
    print 'fetch one record:'
    result = cursor.fetchone()
    print result
    print 'id: %s,name: %s' %(result[0],result[1])
    
    # 查詢多條記錄
    print 'fetch five record:'
    results = cursor.fetchmany(5)
    for r in results:
        print r
        
    # 查詢所有記錄
    # 重置游標(biāo)位置蛤奥,偏移量:大于0向后移動(dòng);小于0向前移動(dòng),mode默認(rèn)是relative
    # relative:表示從當(dāng)前所在的行開(kāi)始移動(dòng); absolute:表示從第一行開(kāi)始移動(dòng)
    cursor.scroll(0,mode='absolute')
    results = cursor.fetchall()
    for r in results:
        print r
 
    cursor.scroll(-2)
    results = cursor.fetchall()
    for r in results:
        print r
 
    # 更新記錄
    cursor.execute('UPDATE %s SET name = "%s" WHERE id = %s' %(TABLE_NAME,'Jack',1))
    # 刪除記錄
    cursor.execute('DELETE FROM %s WHERE id = %s' %(TABLE_NAME,2))
    
    # 如果沒(méi)有設(shè)置自動(dòng)提交事務(wù)纷跛,則這里需要手動(dòng)提交一次
    conn.commit()
except:
    import traceback
    traceback.print_exc()
    # 發(fā)生錯(cuò)誤時(shí)會(huì)滾
    conn.rollback()
finally:
    # 關(guān)閉游標(biāo)連接
    cursor.close()
    # 關(guān)閉數(shù)據(jù)庫(kù)連接
    conn.close()

查詢時(shí)返回字典結(jié)構(gòu)

MySQLdb默認(rèn)查詢結(jié)果都是返回tuple喻括,通過(guò)使用不同的游標(biāo)可以改變輸出格式,這里傳遞一個(gè)cursors.DictCursor參數(shù)贫奠。

import MySQLdb.cursors

conn = MySQLdb.connect(host='localhost', user='root', passwd='root', db='test', cursorclass=MySQLdb.cursors.DictCursor)
cursor = conn.cursor()

cursor.execute('select * from user')
r = cursor.fetchall()
print r
# 當(dāng)使用位置參數(shù)或字典管理參數(shù)時(shí)唬血,必須導(dǎo)入MySQLdb.cursors模塊

# 也可以用下面的寫法
import MySQLdb as mdb
conn  = mdb.connect('localhost', 'root', 'root', 'test')
cursor = conn.cursor(cursorclass=mdb.cursors.DictCursor)

cursor.execute('select * from user')
r = cursor.fetchall()
print r

MySQLdb取回大結(jié)果集的技巧

普通的操作無(wú)論是fetchall()還是fetchone()都是先將數(shù)據(jù)載入到本地再進(jìn)行計(jì)算,大量的數(shù)據(jù)會(huì)導(dǎo)致內(nèi)存資源消耗光唤崭。解決辦法是使用SSCurosr光標(biāo)來(lái)處理拷恨。

然而,在python3下谢肾,MySQLdb模塊不再提供支持腕侄,此時(shí)可以使用另一個(gè)模塊PyMySQL,它支持python2和python3芦疏。

2. PyMySQL

PyMySQL是一個(gè)純Python寫的MySQL客戶端冕杠,它的目標(biāo)是替代MySQLdb,可以在CPython酸茴、PyPy分预、IronPython和Jython環(huán)境下運(yùn)行。PyMySQL在MIT許可下發(fā)布薪捍。

PyMySQL的性能和MySQLdb幾乎相當(dāng)笼痹,如果對(duì)性能要求
不是特別的強(qiáng),使用PyMySQL將更加方便酪穿。

PyMySQL的使用方法和MySQLdb幾乎一樣凳干。

安裝

pip install pymysql

實(shí)例

#!/usr/bin/env python
# -*- coding: utf-8 -*-
import pymysql

config = {
    'host': '127.0.0.1',
    'port': 3306,
    'user': 'root',
    'passwd': 'root',
    'charset':'utf8mb4',
    'cursorclass':pymysql.cursors.DictCursor
    }
conn = pymysql.connect(**config)
conn.autocommit(1)
cursor = conn.cursor()

try:
    # 創(chuàng)建數(shù)據(jù)庫(kù)
    DB_NAME = 'test'
    cursor.execute('DROP DATABASE IF EXISTS %s' %DB_NAME)
    cursor.execute('CREATE DATABASE IF NOT EXISTS %s' %DB_NAME)
    conn.select_db(DB_NAME)
    
    #創(chuàng)建表
    TABLE_NAME = 'user'
    cursor.execute('CREATE TABLE %s(id int primary key,name varchar(30))' %TABLE_NAME)
    
    # 批量插入紀(jì)錄
    values = []
    for i in range(20):
        values.append((i,'kk'+str(i)))
    cursor.executemany('INSERT INTO user values(%s,%s)',values)
    
    # 查詢數(shù)據(jù)條目
    count = cursor.execute('SELECT * FROM %s' %TABLE_NAME)
    print 'total records:', cursor.rowcount
    
    # 獲取表名信息
    desc = cursor.description
    print "%s %3s" % (desc[0][0], desc[1][0])

    cursor.scroll(10,mode='absolute')
    results = cursor.fetchall()
    for result in results:
        print result
        
except:
    import traceback
    traceback.print_exc()
    # 發(fā)生錯(cuò)誤時(shí)會(huì)滾
    conn.rollback()
finally:
    # 關(guān)閉游標(biāo)連接
    cursor.close()
    # 關(guān)閉數(shù)據(jù)庫(kù)連接
    conn.close()

輸出結(jié)果:

total records: 20
id name
{u'id': 10, u'name': u'kk10'}
{u'id': 11, u'name': u'kk11'}
{u'id': 12, u'name': u'kk12'}
{u'id': 13, u'name': u'kk13'}
{u'id': 14, u'name': u'kk14'}
{u'id': 15, u'name': u'kk15'}
{u'id': 16, u'name': u'kk16'}
{u'id': 17, u'name': u'kk17'}
{u'id': 18, u'name': u'kk18'}
{u'id': 19, u'name': u'kk19'}

參考:

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市被济,隨后出現(xiàn)的幾起案子救赐,更是在濱河造成了極大的恐慌,老刑警劉巖只磷,帶你破解...
    沈念sama閱讀 207,248評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件净响,死亡現(xiàn)場(chǎng)離奇詭異少欺,居然都是意外死亡喳瓣,警方通過(guò)查閱死者的電腦和手機(jī)馋贤,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,681評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)畏陕,“玉大人配乓,你說(shuō)我怎么就攤上這事』莼伲” “怎么了犹芹?”我有些...
    開(kāi)封第一講書(shū)人閱讀 153,443評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)鞠绰。 經(jīng)常有香客問(wèn)我腰埂,道長(zhǎng),這世上最難降的妖魔是什么蜈膨? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,475評(píng)論 1 279
  • 正文 為了忘掉前任屿笼,我火速辦了婚禮,結(jié)果婚禮上翁巍,老公的妹妹穿的比我還像新娘驴一。我一直安慰自己,他們只是感情好灶壶,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,458評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布肝断。 她就那樣靜靜地躺著,像睡著了一般驰凛。 火紅的嫁衣襯著肌膚如雪胸懈。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,185評(píng)論 1 284
  • 那天恰响,我揣著相機(jī)與錄音趣钱,去河邊找鬼。 笑死渔隶,一個(gè)胖子當(dāng)著我的面吹牛羔挡,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播间唉,決...
    沈念sama閱讀 38,451評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼绞灼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了呈野?” 一聲冷哼從身側(cè)響起低矮,我...
    開(kāi)封第一講書(shū)人閱讀 37,112評(píng)論 0 261
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎被冒,沒(méi)想到半個(gè)月后军掂,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體轮蜕,經(jīng)...
    沈念sama閱讀 43,609評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,083評(píng)論 2 325
  • 正文 我和宋清朗相戀三年蝗锥,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了跃洛。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,163評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡终议,死狀恐怖汇竭,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情穴张,我是刑警寧澤细燎,帶...
    沈念sama閱讀 33,803評(píng)論 4 323
  • 正文 年R本政府宣布,位于F島的核電站皂甘,受9級(jí)特大地震影響玻驻,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜偿枕,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,357評(píng)論 3 307
  • 文/蒙蒙 一璧瞬、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧益老,春花似錦彪蓬、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,357評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至桃纯,卻和暖如春酷誓,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背态坦。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,590評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工盐数, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人伞梯。 一個(gè)月前我還...
    沈念sama閱讀 45,636評(píng)論 2 355
  • 正文 我出身青樓玫氢,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親谜诫。 傳聞我的和親對(duì)象是個(gè)殘疾皇子漾峡,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,925評(píng)論 2 344

推薦閱讀更多精彩內(nèi)容