背景

sys_stat_activity视图中用两个字段表示:

backend_xid表示事务开始需要申请的事务id

backend_xmin表示一个事务快照,表示当前数据库中最小的正在运行的事务号,这个快照有可能是很久之前申请过的快照号,如果事务不结束,那么这个快照号一直保留,这符合MVCC特性,但是代价是表膨胀,vacuum不能及时回收。

我们可以通过这两个字段判断数据库中是否有长事务:

select * from sys_stat_activity where state<>'idle' and pg_backend_pid() != pid and (backend_xid is not null or backend_xmin is not null ) and extract(epoch from (now() - xact_start))  > 3; <时间阈值,单位秒> ;

下面我们看一下不同隔离级别下xmin的变化情况,借此理解MVCC。

实验

repeatable read隔离级别的事务,事务的第一条SQL会获取快照(xmin),快照持续到事务结束释放。

[](javascript:void(0)

session A:
获取当前会话PID,并开启一个repeatable read隔离级别的事务
test=# select sys_backend_pid();
sys_backend_pid
----------------
25481
(1 row)
test=# begin transaction isolation level repeatable read;
BEGIN session B:
查询会话A的xmin,xid
test=# select pid,backend_xid,backend_xmin from sys_stat_activity where pid=25481;
pid | backend_xid | backend_xmin
-------+-------------+--------------
25481 | |
(1 row) session A:
执行第一条SQL
test=# select 1;
?column?
----------
1
(1 row) session B:
115494为session A事务开启时数据库集群中未分配的最小事务号,或者未结束的最小事务号
test=# select pid,backend_xid,backend_xmin from sys_stat_activity where pid=25481;
pid | backend_xid | backend_xmin
-------+-------------+--------------
25481 | | 115494
(1 row)
在session B消耗2个事务(使用两种方法,各消耗1个事务ID)
test=# select txid_current();
txid_current
--------------
115494
(1 row)
test=# insert into tbl1 values (1);
INSERT 0 1 session A:
执行第二条SQL
test=# select 2;
?column?
----------
2
(1 row) session B:
对于repeatable read隔离级别的事务来说,xmin不会变化。
test=# select pid,backend_xid,backend_xmin from sys_stat_activity where pid=25481;
pid | backend_xid | backend_xmin
-------+-------------+--------------
25481 | | 115494
(1 row) session A:
执行一条会申请XID的SQL,例如插入数据。
test=# insert into tbl1 values (1);
INSERT 0 1 session B:
xid有值了,为session A对应事务申请下来的事务号,是已消耗掉的,XID不会变化。
test=# select pid,backend_xid,backend_xmin from sys_stat_activity where pid=25481;
pid | backend_xid | backend_xmin
-------+-------------+--------------
25481 | 115495| 115494
(1 row)

[](javascript:void(0)

read committed隔离级别的事务,事务的每一条SQL都会获取快照,SQL执行结束就会释放快照。

[](javascript:void(0)

session A:
获取当前会话PID,并开启一个read committed隔离级别的事务
test=# select sys_backend_pid();
sys_backend_pid
----------------
57479
(1 row)
test=# begin transaction isolation level read committed;
BEGIN session B:
查询会话A的xmin,xid
test=# select pid,backend_xid,backend_xmin from sys_stat_activity where pid=57479;
pid | backend_xid | backend_xmin
-------+-------------+--------------
57479 | |
(1 row) session A:
执行第一条SQL
test=# select 1;
?column?
----------
1
(1 row) session B:
并没有观察到xmin,因为select 1;开始时获取,SQL执行结束,xmin马上就释放了
test=# select pid,backend_xid,backend_xmin from sys_stat_activity where pid=57479;
pid | backend_xid | backend_xmin
-------+-------------+--------------
57479 | |
(1 row) session A:
执行第一条long SQL
test=# select sys_sleep(20); session B:
sys_sleep执行结束前查看,可以观察到xmin,SQL执行结束再查看xmin就会消失。
test=# select pid,backend_xid,backend_xmin from sys_stat_activity where pid=57479;
pid | backend_xid | backend_xmin
-------+-------------+--------------
57479 | | 78339
(1 row) session B:
在session B消耗2个事务(使用两种方法,各消耗1个事务ID)
test=# select txid_current();
txid_current
--------------
78339
(1 row)
test=# insert into tbl1 values (1);
INSERT 0 1 session A:
执行第2条long SQL
test=# select sys_sleep(20); session B:
sys_sleep执行结束前查看,可以观察到xmin,SQL执行结束再查看xmin就会消失。
xmin与第一条SQL看到的78339不一样,因为read committed隔离级别的事务,每条SQL开始时都会新申请快照。注意这里和repeatable read隔离级别不一样,它对应的xmin不会变化。
test=# select pid,backend_xid,backend_xmin from sys_stat_activity where pid=57479;
pid | backend_xid | backend_xmin
-------+-------------+--------------
57479 | | 78341
(1 row) session A:
执行一条会申请XID的SQL,例如插入数据。
test=# insert into tbl1 values (1);
INSERT 0 1 session B:
xid有值了,为session A对应事务申请下来的事务号,是已消耗掉的,XID不会变化,直到事务结束(session A commit后,backend_xid:78341变为空)。而backend_xmin为空表示已经没有未结束的最老的快照版本。这是好的现象,这不会引起表膨胀后vacuum进程回收失败。
test=# select pid,backend_xid,backend_xmin from sys_stat_activity where pid=57479;
pid | backend_xid | backend_xmin
-------+-------------+--------------
57479 | 78341 |
(1 row)

[](javascript:void(0)

总结

以上实验帮助大家理解不同隔离级别下,根据pid查出backend_xid,backend_xmin对应的区别变化,更好理解这两个字段的涵义。当这两个字段任何一个有值时都表示有未结束的事务,这也对应起来开头提到的查询长事务的语句中的where条件:backend_xid is not null or backend_xmin is not null。

当然我们最常用的隔离级别是read committed,这是我们KES数据库的默认隔离级别,也是oracle的默认隔离级别。只有一些特殊的业务场景需要用到repeatable read隔离级别。

KingbaseESV8R6不同隔离级下xmin的区别的更多相关文章

  1. python 里面的单下划线与双下划线的区别

    python 里面的单下划线与双下划线的区别 Python 用下划线作为变量前缀和后缀指定特殊变量. _xxx 不能用'from moduleimport *'导入 __xxx__ 系统定义名字 __ ...

  2. 测试Python类成员的单下划线,双下划线,两头下划线的区别

    首先原谅一个菜鸟叫他“两头下划线”.记得在windows编程中,很多宏定义使用下划线+大写,给人逼格很高的错觉.对于Python下划线的认识,大概是从__dict__这个属性开始的,看__dict__ ...

  3. php windows与linux下的路径区别

    php windows与linux下的路径区别windows用的是"\",linux用的是"/"这一点要特别清楚,, ps:在PHP windows也可以用/表 ...

  4. 114、drawable和mipmap 目录下图片的区别

    android 在 API level 17 加入了 mipmap 技术,对 bitmap 图片的渲染支持 mipmap 技术,来提高渲染的速度和质量.mipmap 是一种很早就有的技术了,翻译过来就 ...

  5. C++和C在linux下 和在windows下有什么区别?

    一.函数库的区别 linux下的C函数库和windows下的函数库系统调用的机制不一样,Glibc包含了主要的C库.这个库提供了基本例程,用于分配内存.搜索目录.打开关闭文件.读写文件.字串处理.模式 ...

  6. Python 私有变量中两个下划线 _ _item 与 一个下划线的区别 _item

    python中没有常量的说法, 但是可以通过元组实现一个常量 在python的私有变量中, 存在两个下划线 _ _item 与一个下划线 _item 的区别 前面带两个下划线的私有变量: 只能在本类中 ...

  7. 【HTML&CSS】 第二章:标准模式下的页面与怪异模式下的页面区别

    盒模型 前面提到,盒模型(box mode)是浏览器 Quirks Mode 和 Standards Mode 的主要区别. 描述 对于“盒模型”一词并没有明确的文档定义,它是开发人员描述 CSS 中 ...

  8. linux下 open fopen区别

    open是linux下的底层系统调用函数,fopen与freopen c/c++下的标准I/O库函数,带输入/输出缓冲.linxu下的fopen是open的封装函数,fopen最终还是要调用底层的系统 ...

  9. linux中bin和xbin下可执行程序的区别

    /bin下的都是Linux最基础的,所有用户都可以使用的外部命令 /sbin下的都是只有超级用户root才能使用的.管理Linux系统的外部命令 /usr/bin以及/usr/local/bin下的都 ...

随机推荐

  1. Elasticsearch 在地理信息空间索引的探索和演进

    vivo 互联网服务器团队- Shuai Guangying 本文梳理了Elasticsearch对于数值索引实现方案的升级和优化思考,从2015年至今数值索引的方案经历了多个版本的迭代,实现思路从最 ...

  2. Linux远程连接工具和运行级别

    常用的Linux远程连接工具: xshell MobaXterm windows的命令行工具 Linux的运行级别 linux有七个运行级别 0----所有的服务都不开启,代表的式关机 1---代表的 ...

  3. python小题目练习(三)

    题目:输出1!+2!+3!+--+10!的结果代码实现: # 定义一个函数来递归实现阶乘操作def cycle(num): if num == 1: return 1 else: return num ...

  4. Arrays.asList的使用

    Arrays.asList的作用是将数组转化为list,一般是用于在初始化的时候,设置几个值进去,简化代码,省去add的部分. 示例: List<String> menuList = Ar ...

  5. PTA(BasicLevel)-1014 福尔摩斯的约会

    一.问题描述 大侦探福尔摩斯接到一张奇怪的字条:我们约会吧! 3485djDkxh4hhGE 2984akDfkkkkggEdsb s&hgsfdk d&Hyscvnm.大侦探很快就明 ...

  6. SQLZOO练习(一)SELECT BASICS,SELECT form world

    name continent area population gdp Afghanistan Asia 652230 25500100 20343000000 Albania Europe 28748 ...

  7. Keyboading 思路

    0x01 前置芝士 还是先放个 link 吧. 所需知识点:BFS. 思维难度较高,实现简单. 0x02 题目大意:其实就是给你个图,按顺序走到相应的点,求所需最少步数(走到需要去的点会耗费一次步数) ...

  8. 如何让照片中的人物笑起来?HMS Core视频编辑服务一键微笑功能,让人物笑容更自然

    最近一键"露齿笑"席卷全网,无论是短视频用户还是社交App用户都在使用这项黑科技.当三两好友聚会拍集体照留念时,为了处理个别人的表情"瑕疵",让大家都尽量保持微 ...

  9. YII服务定位器依赖注入

    <?php /** * Created by PhpStorm. * Date: 2016/5/25 * Time: 18:33 * 服务定位器依赖注入 */ namespace fronten ...

  10. 快速新建并配置一个eslint+prettier+husky+commitlint+vue3+vite+ts+pnpm的项目

    前置准备 一台电脑 vscode pnpm vscode插件:ESLint v2.2.6及以上 vscode插件:Prettier - Code formatter v9.5.0及以上 vscode插 ...