背景

客户现场执行压测时候,发生周期性的TPS大幅下降,通过查看kwr报告发现DBcpu时间占DBtime时间很少,百分之90的DBtime花费在tuple锁等待上,等待事件类型是lock。

等待时间最多的语句是select fd_id,ctid,xmin from ... for update

含义

select for update语句目的在查询时,避免其他用户对该表进行dml等操作,造成表的不一致性。

select for update语句需要等待表锁,以及表对应行锁释放之后才能返回查询结果。

表级锁模式

常见锁模式以及应用场景:

ACCESS SHARE:select操作获取该模式锁资源,通常情况下所有只读取不修改表的查询都会获取该模式锁资源

ROW SHARE: select for update 和 select for share 命令获取该模式锁资源

ROW EXCLUSIVE: DML操作通过会获取该模式锁资源,通常情况下任何需要修改表数据的操作都会持有该模式锁资源

SHARE UPDATE EXCLUSIVE:对于 vacuum(非full)、create index concurrnetly、reindex concurrently、create statistics、alter index相关、alter table相关操作会持有该模式锁资源,该模式锁资源主要是为了范围并发对同一张表DDL变更以及vacuum操作

SHARE: create index (非concurrently)操作会持有该模式锁资源,该模式锁资源可阻止并发对表的创建索引操作

SHARE ROW EXCLUSIVE: cerate trigger、一些alter table操作会持有该模式锁资源

EXCLUSIVE: refresh materialized view concurrently操作会持有该模式锁资源

ACCESS EXCLUSIVE:lock table模式锁定模式,drop table、truncate、reindex、 cluster、vacuum full、refresh materialized view (without concurrently) 还有一些alter table、alter index操作需要获取该模式锁资源,该模式锁资源保证了持有者唯一访问表

测试

测试环境隔离级别是 read committed,数据库版本是KingbaseESV8R6C7

1.update语句阻塞select for update语句

开启一个会话,不提交
begin;
update t3 set name='qite' where id=1; 开启另一个会话执行,发现被阻塞
select id,name from t3 for update; 在另一个会话查看锁情况。
查看表t3有关的锁,这里表t3持有了tuple上的AccessExclusiveLock锁,这是最高级别的锁,阻塞一切sql语句。
pid4203持有的锁row share表示正在执行select for update语句,AccessExclusiveLock最高级别锁也是pid4203需要获取的。
pid4205持有的RowExclusiveLock锁表示执行dml语句需要获取的锁,这里可以看出pid4205会话很可能执行了update语句。
SELECT
pg_locks.pid,
a.datname,
locktype,
virtualtransaction,
transactionid,
nspname,
relname,
mode,
granted,
cast(date_trunc('second',query_start) AS timestamp) AS query_start
FROM
pg_locks
LEFT OUTER JOIN pg_class ON (pg_locks.relation = pg_class.oid)
LEFT OUTER JOIN pg_namespace ON (pg_namespace.oid = pg_class.relnamespace),
pg_stat_activity a
WHERE NOT pg_locks.pid = pg_backend_pid()
AND pg_locks.pid=a.pid
and pg_class.relname='t3';
pid | datname | locktype | virtualtransaction | transactionid | nspname | relname | mode | granted | query_start
------+---------+----------+--------------------+---------------+---------+---------+---------------------+---------+---------------------
4203 | test | tuple | 4/43 | | public | t3 | AccessExclusiveLock | t | 2023-05-16 15:39:11
4203 | test | relation | 4/43 | | public | t3 | RowShareLock | t | 2023-05-16 15:39:11
4205 | test | relation | 5/10 | | public | t3 | RowExclusiveLock | t | 2023-05-16 15:39:08
4205 | test | relation | 5/10 | | public | t3 | AccessShareLock | t | 2023-05-16 15:39:08
(4 rows) 执行另一个更直观的sql语句查看锁情况,开启第三个会话执行如下sql。
SELECT blocked_locks.pid AS blocked_pid,
blocked_activity.usename AS blocked_user,
blocking_locks.pid AS blocking_pid,
blocking_activity.usename AS blocking_user,
blocked_activity.query AS blocked_statement,
blocking_activity.query AS current_statement_in_blocking_process
FROM sys_catalog.sys_locks blocked_locks
JOIN sys_catalog.sys_stat_activity blocked_activity ON blocked_activity.pid = blocked_locks.pid
JOIN sys_catalog.sys_locks blocking_locks
ON blocking_locks.locktype = blocked_locks.locktype
AND blocking_locks.DATABASE IS NOT DISTINCT FROM blocked_locks.DATABASE
AND blocking_locks.relation IS NOT DISTINCT FROM blocked_locks.relation
AND blocking_locks.page IS NOT DISTINCT FROM blocked_locks.page
AND blocking_locks.tuple IS NOT DISTINCT FROM blocked_locks.tuple
AND blocking_locks.virtualxid IS NOT DISTINCT FROM blocked_locks.virtualxid
AND blocking_locks.transactionid IS NOT DISTINCT FROM blocked_locks.transactionid
AND blocking_locks.classid IS NOT DISTINCT FROM blocked_locks.classid
AND blocking_locks.objid IS NOT DISTINCT FROM blocked_locks.objid
AND blocking_locks.objsubid IS NOT DISTINCT FROM blocked_locks.objsubid
AND blocking_locks.pid != blocked_locks.pid
JOIN sys_catalog.sys_stat_activity blocking_activity ON blocking_activity.pid = blocking_locks.pid
WHERE NOT blocked_locks.GRANTED;
blocked_pid | blocked_user | blocking_pid | blocking_user | blocked_statement | current_statement_in_blocking_process
-------------+--------------+--------------+---------------+------------------------------------+---------------------------------------
4203 | system | 4205 | system | select id,name from t3 for update; | update t3 set name='qite' where id=1;
(1 row) 在操作系统查看有关pid,4203pid中的select语句被阻塞,阻塞源头是第一个会话执行的update语句,pid是4205。
kingbas+ 4203 4154 0 15:27 ? 00:00:00 kingbase: system test [local] SELECT waiting
kingbas+ 4205 4154 0 15:27 ? 00:00:00 kingbase: system test [local] idle in transaction 第一个会话中的update语句commit或rollback后,事务结束后,第二个会话中的select for update语句可以执行成功,原因是blocking_pid被释放了。

2.select for update语句阻塞update语句

开启一个会话执行如下sql
begin;
select id,name from t3 for update; 开启另一个会话执行,发现被阻塞
update t3 set name='qitu' where id=1; 在另一个会话查看锁情况
pid4203对应RowExclusiveLock锁表示执行dml语句需要获取的锁,pid4205持有的RowShareLock锁说明此会话在执行select for update语句。
SELECT
pg_locks.pid,
a.datname,
locktype,
virtualtransaction,
transactionid,
nspname,
relname,
mode,
granted,
cast(date_trunc('second',query_start) AS timestamp) AS query_start
FROM
pg_locks
LEFT OUTER JOIN pg_class ON (pg_locks.relation = pg_class.oid)
LEFT OUTER JOIN pg_namespace ON (pg_namespace.oid = pg_class.relnamespace),
pg_stat_activity a
WHERE NOT pg_locks.pid = pg_backend_pid()
AND pg_locks.pid=a.pid
and pg_class.relname='t3';
pid | datname | locktype | virtualtransaction | transactionid | nspname | relname | mode | granted | query_start
------+---------+----------+--------------------+---------------+---------+---------+------------------+---------+---------------------
4203 | test | tuple | 4/44 | | public | t3 | ExclusiveLock | t | 2023-05-16 15:54:16
4203 | test | relation | 4/44 | | public | t3 | RowExclusiveLock | t | 2023-05-16 15:54:16
4203 | test | relation | 4/44 | | public | t3 | AccessShareLock | t | 2023-05-16 15:54:16
4205 | test | relation | 5/11 | | public | t3 | RowShareLock | t | 2023-05-16 15:54:00
(4 rows) 执行如下sql,果然阻塞源是pig4205执行的语句select for update。
SELECT blocked_locks.pid AS blocked_pid,
blocked_activity.usename AS blocked_user,
blocking_locks.pid AS blocking_pid,
blocking_activity.usename AS blocking_user,
blocked_activity.query AS blocked_statement,
blocking_activity.query AS current_statement_in_blocking_process
FROM sys_catalog.sys_locks blocked_locks
JOIN sys_catalog.sys_stat_activity blocked_activity ON blocked_activity.pid = blocked_locks.pid
JOIN sys_catalog.sys_locks blocking_locks
ON blocking_locks.locktype = blocked_locks.locktype
AND blocking_locks.DATABASE IS NOT DISTINCT FROM blocked_locks.DATABASE
AND blocking_locks.relation IS NOT DISTINCT FROM blocked_locks.relation
AND blocking_locks.page IS NOT DISTINCT FROM blocked_locks.page
AND blocking_locks.tuple IS NOT DISTINCT FROM blocked_locks.tuple
AND blocking_locks.virtualxid IS NOT DISTINCT FROM blocked_locks.virtualxid
AND blocking_locks.transactionid IS NOT DISTINCT FROM blocked_locks.transactionid
AND blocking_locks.classid IS NOT DISTINCT FROM blocked_locks.classid
AND blocking_locks.objid IS NOT DISTINCT FROM blocked_locks.objid
AND blocking_locks.objsubid IS NOT DISTINCT FROM blocked_locks.objsubid
AND blocking_locks.pid != blocked_locks.pid
JOIN sys_catalog.sys_stat_activity blocking_activity ON blocking_activity.pid = blocking_locks.pid
WHERE NOT blocked_locks.GRANTED;
blocked_pid | blocked_user | blocking_pid | blocking_user | blocked_statement | current_statement_in_blocking_process
-------------+--------------+--------------+---------------+---------------------------------------+---------------------------------------
4203 | system | 4205 | system | update t3 set name='qitu' where id=1; | select id,name from t3 for update;
(1 row) 操作系统查看对应pid情况,pid4203执行update语句被阻塞,阻塞源是pid4205。
kingbas+ 4203 4154 0 15:27 ? 00:00:00 kingbase: system test [local] UPDATE waiting
kingbas+ 4205 4154 0 15:27 ? 00:00:00 kingbase: system test [local] idle in transaction 解决方法是持有锁会话结束事务(提交或回滚),随后被阻塞的update语句执行成功,或杀掉持有锁的进程。

总结

select for update几种其他用法:

select * from t for update nowait 不等待行锁释放,提示锁冲突,不返回结果。

select * from t for update wait 5 等待5秒,若行锁仍未释放,则提示锁冲突,不返回结果。

select * from t for update skip locked 查询返回查询结果,但忽略有行锁的记录。

使用FOR UPDATE WAIT”子句有几个优点:

1.防止无限期地等待被锁定的行。

2.允许应用程序中对锁的等待时间进行控制。

3.在不确定sql执行前表是否被锁,wait关键字起到关键作用。

4.若使用了skip locked,则可以越过锁定的行,而不会报错。

此案例提供了数据库中锁阻塞的排查方法,如生产环境中遇到其他类型lock重型锁阻塞,也可用以上方法分析。

KingbaseES 中select for update语句引起的锁问题的更多相关文章

  1. 数据库中Select For update语句的解析

    ----------- Oracle -----------------– Oracle 的for update行锁 键字: oracle 的for update行锁 SELECT-FOR UPDAT ...

  2. KingbaseES 中select distinct on 语句

    用法 SELECT DISTINCT ON ( expression [, ...] ) 把记录根据[, -]的值进行分组,分组之后仅返回每一组的第一行. 需要注意的是,如果不指定ORDER BY子句 ...

  3. mysql 多列唯一索引在事务中select for update是不是行锁?

    在表中有这么一索引 UNIQUE KEY `customer_id` (`customer_id`,`item_id`,`ref_id`) 问1. 这种多列唯一索引在事务中select for upd ...

  4. Select For update语句浅析 (转)

    Select … for update语句是我们经常使用手工加锁语句.通常情况下,select语句是不会对数据加锁,妨碍影响其他的DML和DDL操作.同时,在多版本一致读机制的支持下,select语句 ...

  5. MySQL中select * for update锁表的范围

    MySQL中select * for update锁表的问题 由于InnoDB预设是Row-Level Lock,所以只有「明确」的指定主键,MySQL才会执行Row lock (只锁住被选取的资料例 ...

  6. MySQL中select * for update锁表的问题

    MySQL中select * for update锁表的问题 由于InnoDB预设是Row-Level Lock,所以只有「明确」的指定主键,MySQL才会执行Row lock (只锁住被选取的资料例 ...

  7. sql server中同时执行select和update语句死锁问题

    原始出处 http://oecpby.blog.51cto.com/2203338/457054 最近在项目中使用SqlServer的时候发现在高并发情况下,频繁更新和频繁查询引发死锁.通常我们知道如 ...

  8. 数据库:Mysql中“select ... for update”排他锁分析

    Mysql InnoDB 排他锁 用法: select … for update; 例如:select * from goods where id = 1 for update; 排他锁的申请前提:没 ...

  9. Select For update语句浅析

    Select -forupdate语句是我们经常使用手工加锁语句.通常情况下,select语句是不会对数据加锁,妨碍影响其他的DML和DDL操作.同时,在多版本一致读机制的支持下,select语句也不 ...

  10. Mysql中“select ... for update”排他锁(转)

    原帖地址 https://blog.csdn.net/claram/article/details/54023216 Mysql InnoDB 排他锁 用法: select … for update; ...

随机推荐

  1. 【Unity3D】GUI控件

    1 前言 ​ Unity 3D 提供了 GUI.NGUI.UGUI 等图形系统,以增强玩家与游戏的交互性.GUI 在编译时不能可视化,在运行时才能可视化.GUI 代码需要在 OnGUI 函数中调用才能 ...

  2. ORA-14550错误解决方法

    工作中修改临时表,报错: ---------------------------------- 以SYSDBA身份登录,执行以下语句: select a.sid, a.serial#,        ...

  3. Vue+SpringBoot+ElementUI实战学生管理系统-1.项目介绍

    1.项目介绍 前段时间有位老铁问老徐有没有Vue+SpringBoot+ElementUI前后分离的项目想学习下,抱歉前端时间有点忙.千呼万唤始出来,做得不是很到位,需要的朋友可以拿去自己定制.:) ...

  4. 解决springboot整合freemarker页面跳转404

    问题说明 spring boot 2.1.5集成freemarker时跳转报404! 集成过程说明 pom.xml <dependency> <groupId>org.free ...

  5. Spring源码之bean的加载

    目录 1. FactoryBean 的使用 2. 缓存中获取单例 bean: 3. 从 bean 实例获取对象, 4. 获取单例 bean (从缓存加载失败): 5. 创建 bean (createB ...

  6. 【Android逆向】修改so文件方式修改程序行为

    1. 还是之前的那个apk 链接:https://pan.baidu.com/s/1vKC1SevvHfeI7f0d2c6IqQ 密码:u1an 尝试使用 010Editor来修改so文件 2. 使用 ...

  7. 基于kubeadm部署k8s1.80.0

    k8s搭建 硬件要求 测试环境 # master 2核 4G 20G # node 4核 8G 40G 生产环境 # master 8核 16G 100G # node 16核 64G 500G 方式 ...

  8. 游戏H5引擎Canvas屏幕自适应CSS代码

    canvas.style = `touch-action: none; width:${ width }px; height:${ height }px; cursor: inherit;`;

  9. ASP.NET Core 从入门到精通-资源收集导航

    ASP.NET Core 从入门到精通-资源收集导航 目录 ASP.NET Core 从入门到精通-资源收集导航 学习路线 学习路线资源导航大全 1,介绍 2,入门 3,教程 创建 Razor 页面 ...

  10. Java 求数值型数组中的最大元素 最小值 平均值 总和等 要求:随机数是 两位数

    1 /* 2 * 3 * 算法考查:求数值型数组中的最大元素 最小值 平均值 总和等 4 * 要求:随机数是 两位数 5 * [10,99] 6 * 公式:(int)(Math.random()*(9 ...