假设这两个表a
,并b
有单排,查询
SELECT * FROM a, b FOR UPDATE
Run Code Online (Sandbox Code Playgroud)
应该得到两个行级锁(一个在a上,一个在b上).是否有任何已定义的锁定获取顺序?并且有没有办法要求表b中的锁从a之前得到锁定(以避免与其他事务的死锁)?
是否有任何已定义的锁定获取顺序?
SELECT *
无论如何,这不是我所知道的.由于没有记录此情况下的锁定顺序,即使在实践中存在,也不能依赖它.它可能会在未来版本中发生变化.
并且有没有办法要求表b中的锁从a之前得到锁定(以避免与其他事务的死锁)?
如果你必须使用SELECT *
那么没有.但是,如果你可以控制SELECT
-list,是的.看起来行锁是按照相关元组字段出现在SELECT
列表中的顺序获取的,因此:
SELECT a.x, b.x FROM b, a FOR UPDATE;
Run Code Online (Sandbox Code Playgroud)
将a
从那时起在该行上获取一个锁b
.目前,无论如何; 我不认为标准指定了这个,并且在文档中没有看到它的任何引用,所以这可能会在以后发生变化.
就个人而言......我会使用一个DO
块或单独的查询.有可能使用一些子查询或CTE来完成它,但您必须在它们之间创建某种形式的人工依赖以确保排序.脆弱而不值得.
让我们看看实际发生了什么:
regress=> EXPLAIN (VERBOSE) SELECT * FROM a, b FOR UPDATE;
QUERY PLAN
-------------------------------------------------------------------------------
LockRows (cost=0.00..129674.00 rows=5760000 width=20)
Output: a.x, b.x, a.ctid, b.ctid
-> Nested Loop (cost=0.00..72074.00 rows=5760000 width=20)
Output: a.x, b.x, a.ctid, b.ctid
-> Seq Scan on public.a (cost=0.00..34.00 rows=2400 width=10)
Output: a.x, a.ctid
-> Materialize (cost=0.00..46.00 rows=2400 width=10)
Output: b.x, b.ctid
-> Seq Scan on public.b (cost=0.00..34.00 rows=2400 width=10)
Output: b.x, b.ctid
(10 rows)
Run Code Online (Sandbox Code Playgroud)
执行查询,然后将结果输入LockRows
节点.怎么LockRows
办?为此,是时候进行源代码潜水了.
ExecLockRows
insrc/backend/executor/nodeLockRows.c
是相关代码.那里有很多,但它的要点是它按顺序迭代RowMark
s 列表并按顺序获取每个锁.该列表由以下内容设置:ExecInitLockRows
复制并过滤在规划期间准备并存储在LockRows
节点中的列表.
我没有时间追溯到规划器中找到LockRows
创建的顺序,但是IIRC它基本上只是解析顺序(for SELECT *
)或字段在SELECT
列表中出现的顺序(如果你没有使用*
).我建议不要依赖它.
归档时间: |
|
查看次数: |
1177 次 |
最近记录: |