亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Python ORM框架SQLAlchemy學(xué)習(xí)筆記之?dāng)?shù)據(jù)添加和事務(wù)回滾介紹

 更新時(shí)間:2014年06月10日 10:52:11   作者:  
這篇文章主要介紹了Python ORM框架SQLAlchemy學(xué)習(xí)筆記之?dāng)?shù)據(jù)添加和事務(wù)回滾介紹,需要的朋友可以參考下

1. 添加一個(gè)新對(duì)象

前面介紹了映射到實(shí)體表的映射類User,如果我們想將其持久化(Persist),那么就需要將這個(gè)由User類建立的對(duì)象實(shí)例添加到我們先前創(chuàng)建的Session會(huì)話實(shí)例中:

復(fù)制代碼 代碼如下:

ed_user = User('ed', 'Ed Jones', 'edspassword')
session.add(ed_user)

上面兩段代碼執(zhí)行完后對(duì)象持久化了么?你或許會(huì)興沖沖的跑去數(shù)據(jù)庫(kù)里查看,結(jié)果卻失望而歸——數(shù)據(jù)庫(kù)里什么都沒有。為什么呢?因?yàn)镾QLAlchemy采取的是Lazyload策略,也就是說現(xiàn)在這個(gè)對(duì)象被標(biāo)記為Pending準(zhǔn)備狀態(tài),但沒有執(zhí)行任何可能導(dǎo)致數(shù)據(jù)庫(kù)變化的SQL語句。那么什么時(shí)候會(huì)執(zhí)行SQL語句并真正持久化呢?這個(gè)要等SQLAlchemy覺得需要的時(shí)候,比如我們現(xiàn)在查詢這個(gè)對(duì)象、對(duì)象的一個(gè)屬性或者顯式的調(diào)用flush方法,這時(shí)候SQLAlchemy覺得它“是時(shí)候”或者“不得不”執(zhí)行SQL數(shù)據(jù)庫(kù)查詢以便于把標(biāo)記為Pending的數(shù)據(jù)寫入數(shù)據(jù)庫(kù)表中了。假如這時(shí)候你執(zhí)行的獲取對(duì)象、對(duì)象屬性或者類似的操作,SQLAlchemy在執(zhí)行完SQL語句后會(huì)將你所要查詢的數(shù)據(jù)反饋給你。


為了更好的說明這一點(diǎn),這里舉一個(gè)例子,這里涉及到我們第一個(gè)查詢示例,我們調(diào)用了Query對(duì)象來幫助我們完成這些,比如這里我們獲取剛剛持久化的用戶ed,我們通過“過濾(filter by)”的方式來查詢用戶名為ed的用戶,當(dāng)然我們只需要一個(gè)ed,假如有多個(gè)重名的ed的話,查詢將會(huì)返回所有叫ed的記錄集列表,我們就選擇第一個(gè)ed吧(first)。

復(fù)制代碼 代碼如下:

>>> our_user = session.query(User).filter_by(name='ed').first()
BEGIN (implicit)
INSERT INTO users (name, fullname, password) VALUES (?, ?, ?)
('ed', 'Ed Jones', 'edspassword')
SELECT users.id AS users_id,
        users.name AS users_name,
        users.fullname AS users_fullname,
        users.password AS users_password
FROM users
WHERE users.name = ?
 LIMIT ? OFFSET ?
('ed', 1, 0)
>>> our_user
<User('ed','Ed Jones', 'edspassword')>

可以看到上面的查詢語句返回了一個(gè)User的實(shí)例,而這個(gè)實(shí)例恰恰是我們先前持久化的。同時(shí)由于我們指定了引擎的echo=True,所以再執(zhí)行查詢時(shí)輸出了SQL語句,我們注意到除了普通的SELECT外,還有額外的INSERT語句,而INSERT處理的就是我們剛剛通過session.add()持久化標(biāo)記為Pending的對(duì)象,也就是說你在實(shí)際操作持久化數(shù)據(jù)時(shí)才會(huì)由延遲加載(lazyload)真正觸發(fā)數(shù)據(jù)庫(kù)操作。

實(shí)際上Session查詢反饋給我們的User對(duì)象和我們剛剛持久化的對(duì)象是同一個(gè)對(duì)象,通過下面的代碼可以檢驗(yàn):

復(fù)制代碼 代碼如下:

>>> ed_user is our_user
True

實(shí)際上這里ORM的操作概念有點(diǎn)類似于標(biāo)識(shí)映射(identity map),也就是說在實(shí)體數(shù)據(jù)庫(kù)之前架設(shè)一張標(biāo)識(shí)映射表,可以看作緩存表的一種,任何存儲(chǔ)數(shù)據(jù)庫(kù)的對(duì)象會(huì)事先停留在這張表上,如果我們要查詢一個(gè)對(duì)象,將事先查詢這張標(biāo)識(shí)映射表,如果這個(gè)對(duì)象存在則直接取出,否則就會(huì)查詢實(shí)體數(shù)據(jù)庫(kù),我覺得這個(gè)有點(diǎn)像緩存的作用,可以這么理解吧。


一旦一個(gè)帶有獨(dú)一無二的主鍵的對(duì)象被Session持久化了,所有使用該主鍵在同一Session上查詢的對(duì)象將是同一個(gè)Python對(duì)象。當(dāng)然對(duì)于在這個(gè)會(huì)話中持久化另外一個(gè)具有相同主鍵的對(duì)象將會(huì)拋出異常錯(cuò)誤(主鍵不能重復(fù))。

如果我們想一次性添加多個(gè)對(duì)象到Session中可以調(diào)用add_all():

復(fù)制代碼 代碼如下:

>>> session.add_all([
...     User('wendy', 'Wendy Williams', 'foobar'),
...     User('mary', 'Mary Contrary', 'xxg527'),
...     User('fred', 'Fred Flinstone', 'blah')])

下面再談?wù)勑薷模偃鏓d覺得他的密碼不太安全,決定修改,可以直接這么做:
復(fù)制代碼 代碼如下:

>>> ed_user.password = 'f8s7ccs'

同樣的道理,這個(gè)改動(dòng)不會(huì)立即反映到數(shù)據(jù)庫(kù)里,當(dāng)然Session意識(shí)到你要修改Ed的密碼,它會(huì)暫時(shí)緩沖這個(gè)改動(dòng),我們可以通過dirty方法了解到我們的改動(dòng):
復(fù)制代碼 代碼如下:

>>> session.dirty
IdentitySet([<User('ed','Ed Jones', 'f8s7ccs')>])

同樣的我們可以通過new方法“窺看”到先前用add_all()持久化的對(duì)象列表:
復(fù)制代碼 代碼如下:

>>> session.new 
IdentitySet([<User('wendy','Wendy Williams', 'foobar')>,
<User('mary','Mary Contrary', 'xxg527')>,
<User('fred','Fred Flinstone', 'blah')>])

當(dāng)然這些改動(dòng)都沒有真正反饋到數(shù)據(jù)庫(kù)里,相當(dāng)于都被ORM緩沖了。接下來我們可以顯式的調(diào)用commit()來告訴Session:“我們目前就添加或者改動(dòng)這么多可以提交數(shù)據(jù)庫(kù)了”:
復(fù)制代碼 代碼如下:

>>> session.commit()
UPDATE users SET password=? WHERE users.id = ?
('f8s7ccs', 1)
INSERT INTO users (name, fullname, password) VALUES (?, ?, ?)
('wendy', 'Wendy Williams', 'foobar')
INSERT INTO users (name, fullname, password) VALUES (?, ?, ?)
('mary', 'Mary Contrary', 'xxg527')
INSERT INTO users (name, fullname, password) VALUES (?, ?, ?)
('fred', 'Fred Flinstone', 'blah')
COMMIT

于是剛才緩沖的數(shù)據(jù)或者變更全部被作為事務(wù)一次性flush到數(shù)據(jù)庫(kù)了,通過輸出的SQL語句我們也可以看出來。

這個(gè)操作完成后被會(huì)話(Session)引用的數(shù)據(jù)庫(kù)連接資源將被回收到連接池中,接下來的對(duì)于這個(gè)Session的任何操作將會(huì)觸發(fā)一個(gè)新的事務(wù)(Transaction),當(dāng)然會(huì)再次和連接池申請(qǐng)獲得數(shù)據(jù)庫(kù)連接資源。

之前文章介紹到Ed的User對(duì)象的id為None,現(xiàn)在讓我們來看看吧:

復(fù)制代碼 代碼如下:

>>> ed_user.id
BEGIN (implicit)
SELECT users.id AS users_id,
        users.name AS users_name,
        users.fullname AS users_fullname,
        users.password AS users_password
FROM users
WHERE users.id = ?
(1,)
1

除了由于echo=True導(dǎo)致輸出的SQL語句,看看是不是有了值,值為1。

無論是立即(commit、flush)或者通過“首次訪問加載(load-on-first-access)”,在Session在數(shù)據(jù)庫(kù)插入一條新記錄后,所有新生成的標(biāo)識(shí)和數(shù)據(jù)庫(kù)生成的默認(rèn)值對(duì)于實(shí)例來說才可以被訪問到。

當(dāng)調(diào)用了commit()以后,SQLAlchemy將會(huì)刷新當(dāng)前事務(wù)的所有數(shù)據(jù)到數(shù)據(jù)庫(kù)里。

2. 事務(wù)回滾

本文以及同系列的文章是以自己的想法翻譯的,不當(dāng)之處還請(qǐng)指正,不做權(quán)威依據(jù)。好了,下面我還是簡(jiǎn)單介紹一下事務(wù)回滾吧,其實(shí)這個(gè)和數(shù)據(jù)庫(kù)的事務(wù)回滾一個(gè)意思,就是我們做錯(cuò)事后要撤消之前的變更。

因?yàn)镾ession是作為事務(wù)(transaction)來工作的,所以我們可以回滾(roll back)先前所做的更改。接下來讓我們做兩個(gè)稍后會(huì)被撤銷(回滾)的更改,第一個(gè)是修改ed_user.name:

復(fù)制代碼 代碼如下:

>>> ed_user.name = 'Edwardo'

第二個(gè)是增加一個(gè)“不期望”的用戶fake_user:
復(fù)制代碼 代碼如下:

>>> fake_user = User('fakeuser', 'Invalid', '12345')
>>> session.add(fake_user)

查詢當(dāng)前會(huì)話,我們可以看到這兩個(gè)變更已經(jīng)被flush到當(dāng)前事務(wù)里了:
復(fù)制代碼 代碼如下:

>>> session.query(User).filter(User.name.in_(['Edwardo', 'fakeuser'])).all()
UPDATE users SET name=? WHERE users.id = ?
('Edwardo', 1)
INSERT INTO users (name, fullname, password) VALUES (?, ?, ?)
('fakeuser', 'Invalid', '12345')
SELECT users.id AS users_id,
        users.name AS users_name,
        users.fullname AS users_fullname,
        users.password AS users_password
FROM users
WHERE users.name IN (?, ?)
('Edwardo', 'fakeuser')
[<User('Edwardo','Ed Jones', 'f8s7ccs')>, <User('fakeuser','Invalid', '12345')>]

好吧,接下來是見證奇跡的時(shí)刻,我們回滾(rolling back)事務(wù):
復(fù)制代碼 代碼如下:

>>> session.rollback()
ROLLBACK
>>> ed_user.name
BEGIN (implicit)
SELECT users.id AS users_id,
        users.name AS users_name,
        users.fullname AS users_fullname,
        users.password AS users_password
FROM users
WHERE users.id = ?
(1,)
u'ed'
>>> fake_user in session
False


我們可以看到ed_user的名字變回ed,并且我們不期望的用戶fake_user被“踢出”會(huì)話(Session)了。

最后,我們可以查詢一下用戶名在['ed', 'fakeuser']范圍的用戶,確保我們的更改是有效的:

復(fù)制代碼 代碼如下:

>>> session.query(User).filter(User.name.in_(['ed', 'fakeuser'])).all()
SELECT users.id AS users_id,
        users.name AS users_name,
        users.fullname AS users_fullname,
        users.password AS users_password
FROM users
WHERE users.name IN (?, ?)
('ed', 'fakeuser')
[<User('ed','Ed Jones', 'f8s7ccs')>]

好了,今天就到這里,今天我們講解了添加對(duì)象和事務(wù)回滾,或多或少穿插了些簡(jiǎn)單的查詢,接下來我們會(huì)介紹較為復(fù)雜一些的查詢語句,敬請(qǐng)期待!

相關(guān)文章

  • Python讀取postgresql數(shù)據(jù)庫(kù)詳情

    Python讀取postgresql數(shù)據(jù)庫(kù)詳情

    這篇文章主要介紹了Python讀取postgresql數(shù)據(jù)庫(kù)詳情,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下
    2022-09-09
  • Python中用表格格式打印列表的兩種實(shí)現(xiàn)

    Python中用表格格式打印列表的兩種實(shí)現(xiàn)

    本文將詳細(xì)介紹如何在 Python 中以表格格式打印列表,以便更好地展示和呈現(xiàn)數(shù)據(jù),文中通過示例代碼介紹的非常詳細(xì),需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • Python中的枚舉函數(shù)enumerate()的具體用法

    Python中的枚舉函數(shù)enumerate()的具體用法

    本文主要介紹了Python中的枚舉函數(shù)enumerate()的具體用法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-06-06
  • 20行代碼教你用python給證件照換底色的方法示例

    20行代碼教你用python給證件照換底色的方法示例

    這篇文章主要介紹了20行代碼教你用python給證件照換底色的方法示例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-02-02
  • 使用pandas或numpy處理數(shù)據(jù)中的空值(np.isnan()/pd.isnull())

    使用pandas或numpy處理數(shù)據(jù)中的空值(np.isnan()/pd.isnull())

    這篇文章主要介紹了使用pandas或numpy處理數(shù)據(jù)中的空值(np.isnan()/pd.isnull()),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-05-05
  • Python加密模塊的hashlib,hmac模塊使用解析

    Python加密模塊的hashlib,hmac模塊使用解析

    這篇文章主要介紹了Python加密模塊的hashlib,hmac模塊使用解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-01-01
  • python實(shí)現(xiàn)搜索文本文件內(nèi)容腳本

    python實(shí)現(xiàn)搜索文本文件內(nèi)容腳本

    這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)搜索文本文件內(nèi)容的腳本,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2018-06-06
  • python base64 decode incorrect padding錯(cuò)誤解決方法

    python base64 decode incorrect padding錯(cuò)誤解決方法

    這篇文章主要介紹了python base64 decode incorrect padding錯(cuò)誤解決方法,本文使用把string補(bǔ)齊等號(hào)的方法解決了這個(gè)錯(cuò)誤,需要的朋友可以參考下
    2015-01-01
  • python實(shí)現(xiàn)ModBusTCP協(xié)議的client功能

    python實(shí)現(xiàn)ModBusTCP協(xié)議的client功能

    Modbus TCP 是一種基于 TCP/IP 協(xié)議棧的 Modbus 通信協(xié)議,它用于在工業(yè)自動(dòng)化系統(tǒng)中進(jìn)行設(shè)備之間的通信,只要通過pymodbus或pyModbusTCP任意模塊就可以實(shí)現(xiàn),本文采用pymodbus,感興趣的朋友跟隨小編一起看看吧
    2023-10-10
  • 對(duì)python特殊函數(shù) __call__()的使用詳解

    對(duì)python特殊函數(shù) __call__()的使用詳解

    今天小編就為大家分享一篇對(duì)python特殊函數(shù) __call__()的使用詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2019-07-07

最新評(píng)論