Flask-SQLAlchemy CRUD

當(dāng)我們習(xí)慣 Flask-SQLAlchemy 對(duì)數(shù)據(jù)庫進(jìn)行 CRUD 操作后荞估,就很難再回到基于原生 SQL 的代碼編寫了杈抢。

我們先來看 Read 操作。用實(shí)際的例子來說明用法纸巷,數(shù)據(jù)還是之前經(jīng)常用到的 emp_master 棒掠,
基于上一篇 Flask 工程的文件結(jié)構(gòu)者祖,在 models.py 中定義的 Class 如下:

class EmpMaster(db.Model):
    emp_id = db.Column(db.Integer, primary_key=True)
    gender = db.Column(db.String(10), nullable=False)
    age = db.Column(db.Integer)
    email = db.Column(db.String(50))
    phone_nr = db.Column(db.String(20))
    education = db.Column(db.String(20))
    marital_stat = db.Column(db.String(20))
    nr_of_children = db.Column(db.Integer)

    def __repr__(self):
        return '<emp_master %r>' % 'id:{}'.format(self.emp_id)

查詢數(shù)據(jù)

查詢所有數(shù)據(jù)

from app.models import db, EmpMaster

print (EmpMaster.query.all())

EmpMaster.query 得到的是 BaseQuery 對(duì)象的實(shí)例立莉,對(duì)于 BaseQuery 類,SQLAlchemy 提供了一些查詢方法用于獲得你想要的查詢對(duì)象七问,比如 all() 方法以 list (列表) 的形式返回所有的查詢結(jié)果蜓耻。

根據(jù)主鍵查詢數(shù)據(jù)

get()get_or_404() 方法根據(jù)主鍵來查詢數(shù)據(jù)。get() 方法返回 model 的實(shí)例械巡,如果找不到刹淌,則返回 None;而 get_or_404() 則返回 Not found 的錯(cuò)誤響應(yīng)讥耗。

EmpMaster.query.get(1001)       # 1001 是主鍵

篩選/過濾數(shù)據(jù)

篩選過濾數(shù)據(jù)通過 filter_by()filter() 方法有勾。假設(shè)我們想查詢所有女性雇員,可以這樣做:

emps = EmpMaster.query.filter_by(gender='Female').all()
if len(emps) > 0:
    for emp in emps:
        print(emp.emp_id, emp.gender, emp.email, emp.education)
else:
    print ('No data.')

filter_by() 方法返回的是 BaseQuery 對(duì)象實(shí)例古程,參數(shù)是關(guān)鍵字表達(dá)式 (keyword expression)蔼卡,可以多個(gè)字段,多個(gè)字段之間為 and 關(guān)系挣磨。比如要查詢所有女性雇逞,并且學(xué)歷為 Bachelor 的員工:

emps = EmpMaster.query.filter_by(gender='Female', education='Bachelor11').all()
if len(emps) > 0:
    for emp in emps:
        print(emp.emp_id, emp.gender, emp.email, emp.education)
else:
    print ('No data.')

更為靈活的方式是通過 filter() 方法,與 filter_by() 方法不同的是茁裙,filter() 方法的參數(shù)是 SQL Expression塘砸。舉幾個(gè)例子來說明。

查找所有 education 為 Bachelor 的員工:

emps = EmpMaster.query.filter(EmpMaster.education == 'Bachelor').all()
for emp in emps:
    print (emp)

如果用 filter_by() 方法晤锥,是這樣的:

emps = EmpMaster.query.filter_by(education='Bachelor').all()

看出區(qū)別來了吧掉蔬。下表列示了常見條件的寫法廊宪。

條件 示例
等于 query.filter(EmpMaster.education == 'Bachelor')
不等于 query.filter(EmpMaster.education != 'Bachelor')
LIKE query.filter(EmpMaster.email.like('s%'))
IN query.filter(EmpMaster.education.in_(['Master', 'Bachelor']))
NOT IN query.filter(~EmpMaster.education.in_(['Master', 'Bachelor']))
IS NULL query.filter(EmpMaster.education == None)
IS NOT NULL query.filter(EmpMaster.education != None)

AND 條件:

EmpMaster.query.filter(EmpMaster.gender=='Female', EmpMaster.education=='Bachelor')

# 或者
from sqlalchemy import and_
emps = EmpMaster.query.filter(
    and_(EmpMaster.gender=='Female', EmpMaster.education=='Bachelor')).all()

OR 條件

from sqlalchemy import or_
emps = EmpMaster.query.filter(
  or_(EmpMaster.marital_stat=='Single', EmpMaster.nr_of_children==0))

Create

SQLAlchemy 有一個(gè) session 對(duì)象,代表臨時(shí)存儲(chǔ)區(qū)女轿,數(shù)據(jù)的變動(dòng)提交到 session 后挤忙,需要調(diào)用 commit() 將修改提交到數(shù)據(jù)庫,或者 rollback() 撤銷未提交的修改谈喳。

from app.models import db, EmpMaster
from app import create_app

app = create_app()

new_emp = EmpMaster(
    emp_id = 9001,
    gender = 'Female',
    age = 18,
    email = 'test@qq.com',
    phone_nr = '13800138000',
    education = 'Master',
    marital_stat = 'Single',
    nr_of_children = 0
)

db.session.add(new_emp)
db.session.commit()

可以使用 add_all() 方法批量創(chuàng)建記錄:

emp1 = EmpMaster(...)
emp2 = EmpMaster(...)

db.session.add_all([emp1, emp2])
db.session.commit()

Update

Update 記錄需要先定位到某記錄,然后對(duì)需要的字段賦值后戈泼,然后調(diào)用 commit() 方法提交修改:

emp = EmpMaster.query.get(9001)
emp.email = 'e9001@qq.com'
db.session.commit()

Delete

Delete 記錄先定位到某記錄婿禽,然后調(diào)用 delete() 方法發(fā)送刪除指到 session,最后調(diào)用 commit() 將請(qǐng)求提交到數(shù)據(jù)庫大猛。

emp = EmpMaster.query.get(9001)
db.session.delete(emp)
db.session.commit()
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末扭倾,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子挽绩,更是在濱河造成了極大的恐慌膛壹,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,204評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件唉堪,死亡現(xiàn)場(chǎng)離奇詭異模聋,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)唠亚,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,091評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門链方,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人灶搜,你說我怎么就攤上這事祟蚀。” “怎么了割卖?”我有些...
    開封第一講書人閱讀 164,548評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵前酿,是天一觀的道長。 經(jīng)常有香客問我鹏溯,道長罢维,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,657評(píng)論 1 293
  • 正文 為了忘掉前任丙挽,我火速辦了婚禮言津,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘取试。我一直安慰自己悬槽,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,689評(píng)論 6 392
  • 文/花漫 我一把揭開白布瞬浓。 她就那樣靜靜地躺著初婆,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上磅叛,一...
    開封第一講書人閱讀 51,554評(píng)論 1 305
  • 那天屑咳,我揣著相機(jī)與錄音,去河邊找鬼弊琴。 笑死兆龙,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的敲董。 我是一名探鬼主播紫皇,決...
    沈念sama閱讀 40,302評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼腋寨!你這毒婦竟也來了聪铺?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,216評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤萄窜,失蹤者是張志新(化名)和其女友劉穎铃剔,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體查刻,經(jīng)...
    沈念sama閱讀 45,661評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡键兜,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,851評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了穗泵。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蝶押。...
    茶點(diǎn)故事閱讀 39,977評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖火欧,靈堂內(nèi)的尸體忽然破棺而出棋电,到底是詐尸還是另有隱情,我是刑警寧澤苇侵,帶...
    沈念sama閱讀 35,697評(píng)論 5 347
  • 正文 年R本政府宣布赶盔,位于F島的核電站,受9級(jí)特大地震影響榆浓,放射性物質(zhì)發(fā)生泄漏于未。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,306評(píng)論 3 330
  • 文/蒙蒙 一陡鹃、第九天 我趴在偏房一處隱蔽的房頂上張望烘浦。 院中可真熱鬧,春花似錦萍鲸、人聲如沸闷叉。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,898評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽握侧。三九已至蚯瞧,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間品擎,已是汗流浹背埋合。 一陣腳步聲響...
    開封第一講書人閱讀 33,019評(píng)論 1 270
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留萄传,地道東北人甚颂。 一個(gè)月前我還...
    沈念sama閱讀 48,138評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像秀菱,于是被迫代替她去往敵國和親振诬。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,927評(píng)論 2 355

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