在我們做PG遷移的時候,用AWS工具經(jīng)常會遇到這樣的提示“Transactioncontrol is only possible from the top level or nested callinvocations without any other intervening command”
在點一下具體的問題,如上圖所示,發(fā)現(xiàn)是commit的問題。
這里我寫了一個存儲過程,用python程序調(diào)用來測試了一下這個問題。
1.首先創(chuàng)建一個表
create tabletest(a1,a2) as select 1 n,current_timestamp t;
2.接下來創(chuàng)建一個存儲過程
create or replaceprocedure test_insert(a1 int, a2 int) as
$$
begin
for i in a1..a2loop
insert into testvalues (i,current_timestamp);
commit;
end loop;
end;
$$ language plpgsql;
3.在命令行執(zhí)行調(diào)用
存儲過程執(zhí)行成功,插入4條數(shù)據(jù)。
4.接下來我們使用python程序來調(diào)用存儲過程
Python程序代碼如下:
執(zhí)行python
發(fā)現(xiàn)報錯,這里報“invalidtransaction termination”,然后指向存儲過程第5行COMMIT。
5.取消存儲過程中的commit
這一次把commit取消,再次執(zhí)行python
查詢數(shù)據(jù)庫中的數(shù)據(jù)
通過測試可以發(fā)現(xiàn),在命令行中不管你在存儲過程中加commit或者不加commit,都不會報錯。而通過外部程序調(diào)用的情況,如果你在存儲過程中加了commit,就會報“invalidtransaction termination”
通過搜索,我們發(fā)現(xiàn)官方文檔如下描述。
If CALL isexecuted in a transaction block, then the called procedure cannotexecute transaction control statements. Transaction controlstatements are only allowed if CALL isexecuted in its own transaction.
如果CALL在事務(wù)塊中執(zhí)行,則被調(diào)用的存儲過程無法執(zhí)行事務(wù)控制語句(也就是commit/rollback)等TCL語句。只有CALL在自己事務(wù)中執(zhí)行時,才允許事務(wù)控制語句。而我們使用python程序模塊psycopg連接的時候,通常是以begin開始運行的,這就代表了CALL在事務(wù)塊中運行,是沒辦法在存儲過程中執(zhí)行commit的。
這一點我們可以再證明一下。再次在存儲過程中增加commit語句,然后運行starttransaction。
可以看到命令行也報了ERROR: invalid transaction termination的錯誤。
那么這個問題如何解決呢?
需要我們在應(yīng)用程序進行設(shè)置,增加conn.autocommit= True,這樣就使用了數(shù)據(jù)庫call中本身的事務(wù),而不是程序在開啟的一層事務(wù)。
還需說明一下,在官方文檔是如下介紹的:
Transactioncontrol is only possible in CALL or DO invocationsfrom the top level or nested CALL or DO invocationswithout any other intervening command. For example, if the call stackis CALLproc1() → CALLproc2() → CALLproc3(),then the second and third procedures can perform transaction controlactions. But if the call stack is CALLproc1() → SELECTfunc2() → CALLproc3(),then the last procedure cannot do transaction control, because ofthe SELECT inbetween.
意思是事務(wù)控制只能在call或者do從頂層進行調(diào)用。在沒有任何其他中間命令的嵌套CALL或DO調(diào)用中也能進行事務(wù)控制。例如,如果調(diào)用棧是CALLproc1() → CALL proc2() → CALLproc3(),那么第二個和第三個過程可以執(zhí)行事務(wù)控制動作。但是如果調(diào)用棧是CALLproc1() → SELECT func2() → CALLproc3(),則最后一個過程不能做事務(wù)控制,因為中間有個SELECT。
以上是我們關(guān)于此類事務(wù)控制問題的一些小研究。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://specialneedsforspecialkids.com/yun/130157.html
摘要:深入解析系列之并發(fā)控制與事務(wù)機制并發(fā)控制旨在針對數(shù)據(jù)庫中對事務(wù)并行的場景,保證中的一致性與隔離。啟動并執(zhí)行第一個命令。事務(wù)管理器分配,并返回事務(wù)快照,因為正在進行中。意味著該行由另一個并發(fā)事務(wù)更新,并且其事務(wù)尚未終止。 showImg(https://segmentfault.com/img/remote/1460000018081793); 深入解析 PostgreSQL 系列整理...
摘要:深入解析系列之并發(fā)控制與事務(wù)機制并發(fā)控制旨在針對數(shù)據(jù)庫中對事務(wù)并行的場景,保證中的一致性與隔離。啟動并執(zhí)行第一個命令。事務(wù)管理器分配,并返回事務(wù)快照,因為正在進行中。意味著該行由另一個并發(fā)事務(wù)更新,并且其事務(wù)尚未終止。 showImg(https://segmentfault.com/img/remote/1460000018081793); 深入解析 PostgreSQL 系列整理...
摘要:以上出自發(fā)行說明,這段指出版本支持自動查殺超過指定時間的空閑事務(wù)連接,下面演示下。修改以下參數(shù)備注參數(shù)單位為毫秒,這里設(shè)置超時空閑事務(wù)時間為秒。數(shù)據(jù)庫日志備注數(shù)據(jù)庫日志里清晰地記錄了進程的連接由于空閑事務(wù)超時被斷開連接。 熟悉 PostgreSQL 的朋友應(yīng)該知道 idle in transaction 進程,引發(fā) idle in transaction 的原因很多,例如應(yīng)用代碼中忘記...
摘要:事務(wù)原子性以下轉(zhuǎn)賬給要么都成功要么失敗標(biāo)識符如果打了雙引號整個引號內(nèi)視為標(biāo)識符都會被理解為小寫比如標(biāo)識符其實是被解釋為同一個但是因此建議是總是打引號中的常量用單引號引用不支持的轉(zhuǎn)義即不能出現(xiàn)字符不代表不能使用四位十六進制六位十六進制如 database cluster: a collection of databases managed by a single PostgreSQL s...
閱讀 1346·2023-01-11 13:20
閱讀 1684·2023-01-11 13:20
閱讀 1132·2023-01-11 13:20
閱讀 1858·2023-01-11 13:20
閱讀 4100·2023-01-11 13:20
閱讀 2704·2023-01-11 13:20
閱讀 1385·2023-01-11 13:20
閱讀 3597·2023-01-11 13:20