背景

昨天遇到客户现场的一个有关复制冲突的问题

备库报错:ERROR: canceling statement due to conflict with recovery,user was holding a relation lock for too long

现场情景是备库执行逻辑备份过程中出现的报错,逻辑备份相当于备库查询语句,snapshot,这时主库业务繁忙,对备库查询对象加上exclusiveAccess锁,相关wal日志传输到备库和备库的查询或逻辑复制产生冲突。

查看pg_stat_database_conflicts视图,冲突类型为cflict_lock,也就是锁冲突。

备库基于参数max_standby_streaming_delay和max_standby _archive_delay的值来解决复制冲突。max_standby_streaming_delay参数确定备库在取消与即将应用的WAL日志冲突的查询之前必须等待多长时间。如果冲突语句在这段时间后仍在运行,那么数据库会取消该语句并发出以下错误消息:

ERROR: canceling statement due to conflict with recovery

然而,此错误信息通常因为备库运行长时间的查询导致。

示例

主库执行DROP,请该语句存储在WAL文件中,稍后会在备库上应用,以保持一致性。一个SELECT语句已经在备库上运行,其运行时间大于max_standby_streaming_delay中的值,然后SELECT语句被取消,以便可以应用wal日志中的DROP语句。

注意备库查询的表要足够大,时间足够长,才可以有时间模拟出冲突的出现

备库运行长查询:
select count(*) from t5; 备库查询运行结束前,在主库操作:
drop table t5;
这时在备库收到如下报错:
ERROR: canceling statement due to conflict with recovery
DETAIL: User was holding a relation lock for too long.
STATEMENT: select * from t5;

此外,当备库上的事务正在读取主库上删除的元组时,可能会发生查询冲突。删除元组,然后在主库上触发autovacuum,会导致与仍在备库上运行的SELECT查询发生冲突;在这种情况下,对备库上的SELECT查询将终止,并显示以下错误消息:

ERROR: canceling statement due to conflict with recovery
DETAIL: User query might have needed to see row versions that must be removed.

这种情况可以设置hot_standby_feedback=on 规避

解决方案

在备库上设置参数max_standby_streaming_delay max_standby_archive_delay 可以使用这些参数来留出更多查询时间,超时后再取消与即将应用的WAL日志冲突的备库的查询语句。这些参数表示从主实例接收到数据后,允许应用WAL日志的总时间。这些参数的指定取决于WAL日志的读取位置,如果从流复制中读取WAL数据,则使用max_standby_streaming_delay参数;如果WAL日志是从存储中的归档位置读取的,则使用max_standby_archive_delay参数。

设置这些参数时,请记住以下几点:

1.如果将这些参数的值设置为-1,则允许备库永远等待冲突查询完成,但是增加复制延迟,意味着备库查询到的数据可能不是最新的。

2.如果将这些参数的值设置为0,则备库的冲突查询将被取消,WAL日志条目将应用于备库。

3.参数的默认值设置为30秒,(如果没有指定单位,则默认毫秒)意味着备库查询等待30s。

根据现场环境调整这些参数的值,以平衡查询取消和复制延迟。

注意:如果要增加max_standby_archive_delay以避免取消与应用WAL日志冲突的查询,那么也考虑增加max_standby_streaming_delay。

如果看到如下错误:

"canceling statement due to conflict with recover" with "DETAIL: User query might have needed to see row versions that must be removed"

这时设置hot_standby_feedback参数,如果激活此参数,如果备库进行长查询,则会从备库发送反馈消息给主库,其中包含最旧活动事务的信息;因此,主库做update操作,同时要保留备库正常查询的这些垃圾版本。(与vacuum_defer_cleanup_age效果类似)

代价1,主库膨胀,因为垃圾版本延迟回收。

代价2,重复扫描垃圾版本,重复耗费垃圾回收进程占用CPU资源。(n_dead_tup会一直处于超过垃圾回收阈值状态,从而autovacuum不断唤醒worker进行回收动作,做无用功)

当主库的 autovacuum_naptime很小,同时autovacuum_vacuum_scale_factor很小时,尤为明显。

代价3,如果期间发生大量垃圾,垃圾版本可能会在事务完成后,集中爆炸性的被回收,产生大量的WAL日志,从而造成WAL写的IO高峰。

你还可以检查备库上的pg_stat_database_conflicts视图,获得由于与备库上恢复冲突而取消的语句的统计信息。

注意:此视图并不显示所有已发生的复制冲突,它只显示导致备库上的查询被取消的那些复制冲突信息。

总结

复制冲突类型有多种,此次介绍的冲突种类为锁复制冲突。

解决wal日志无法apply有两个策略:

  1. 备库告诉主库它的查询需要哪些版本,让主库保留,备库查询始终能拿到需要的版本。实际不阻塞apply wal日志,可以设置参数hot_standby_feedback。
  2. 备库wal日志apply进入等待,直到备库冲突查询结束,继续apply,可以设置参数max_standby_streaming_delay。

这两个参数原理近似,区别是hot_standby_feedback可以保证备库总能实现长查询不产生冲突,他主要解决快照冲突类型,而不是锁冲突类型;而max_standby_streaming_delay这个参数设置时间短于备库查询时间,超过此参数时间后,查询被取消,所以这个延迟wal日志参数设置多久需要根据实际场景设置。

KingbaseES V8R6 复制冲突之锁类型冲突的更多相关文章

  1. 关于"作数类型冲突: nvarchar 与 image 不兼容"的问题

    数据库如果是Image类型,当执行插入语句时,如果插入的值是DBNull.Value时提示:操作数类型冲突: nvarchar 与 image 不兼容; 出现这个问题的原因是没有指定DbType的原因 ...

  2. mysql 主从,主主,主主复制时的主键冲突解决

    原理:slave 的i/o thread ,不断的去master抓取 bin_log, 写入到本地relay_log 然后sql thread不断的更新slave的数据 把主服务器所有的数据复制给从服 ...

  3. KingbaseES V8R6集群部署案例之---Windows环境配置主备流复制(异机复制)

    案例说明: 目前KingbaseES V8R6的Windows版本不支持数据库sys_rman的物理备份,可以考虑通过建立主备流复制实现数据库的异机物理备份.本案例详细介绍了,在Windows环境下建 ...

  4. KingbaseES V8R6集群部署案例之---Windows环境配置主备流复制(同一主机)

    案例说明: 目前KingbaseES V8R6的Windows版本不支持数据库sys_rman的物理备份,可以考虑通过建立主备流复制实现数据库的异机物理备份.本案例详细介绍了,在Windows环境下建 ...

  5. SQL Server锁类型

    SQL Server锁类型(SQL)收藏 1. HOLDLOCK: 在该表上保持共享锁,直到整个事务结束,而不是在语句执行完立即释放所添加的锁. 2. NOLOCK:不添加共享锁和排它锁,当这个选项生 ...

  6. java锁类型

    转载链接在每个锁类型后边 线程锁类型 1.自旋锁 ,自旋,jvm默认是10次吧,有jvm自己控制.for去争取锁 锁作为并发共享数据,保证一致性的工具,在JAVA平台有多种实现(如 synchroni ...

  7. sql 锁类型与锁机制 转

      SQL Server锁类型(SQL)收藏1. HOLDLOCK: 在该表上保持共享锁,直到整个事务结束,而不是在语句执行完立即释放所添加的锁.   2. NOLOCK:不添加共享锁和排它锁,当这个 ...

  8. sql 锁类型与锁机制

    SQL Server锁类型(SQL)收藏1. HOLDLOCK: 在该表上保持共享锁,直到整个事务结束,而不是在语句执行完立即释放所添加的锁.   2. NOLOCK:不添加共享锁和排它锁,当这个选项 ...

  9. Oracle数据库的锁类型

    Oracle数据库的锁类型 博客分类: oracle   Oracle数据库的锁类型 根据保护的对象不同,Oracle数据库锁可以分为以下几大类:DML锁(data   locks,数据锁),用于保护 ...

  10. MySQL数据库锁类型

    锁概念 : 当高并发访问同一个资源时,可能会导致数据不一致,需要一种机制将用户访问数据的顺序进行规范化,以保证数据库数据的一致性.锁就是其中的一种机制. 一个栗子 :以买火车票为例,火车票可面向广大消 ...

随机推荐

  1. 基于keras的卷积神经网络(CNN)

    1 前言 本文以MNIST手写数字分类为例,讲解使用一维卷积和二维卷积实现 CNN 模型.关于 MNIST 数据集的说明,见使用TensorFlow实现MNIST数据集分类.实验中主要用到 Conv1 ...

  2. Swoole从入门到入土(22)——多进程[Process]

    Swoole中的Process模块比原生php提供的pcntl模块,提供了更易用的多进程编程接口. 简单总结,Process模块有如下特点: · 可以方便的实现进程间通讯· 支持重定向标准输入和输出, ...

  3. spring boot与junit集成测试

    先创建一个REST接口 package com.laoxu.gamedog.controller; import org.springframework.web.bind.annotation.Req ...

  4. win32 - 虚拟内存的一些介绍

    对32位Windows来说,其虚拟地址空间总数就是2的32次方,即4GB. 如果没有在引导时加上/3GB或/BOOTVA选项,Windows默认最大会分2GB给内核模式程序使用,2GB给用户模式程序. ...

  5. 【LeetCode剑指offer 01】数组中重复的数字、两个栈实现队列

    数组中重复的数字 数组中重复的数字 找出数组中重复的数字. 在一个长度为 n 的数组 nums 里的所有数字都在 0-n-1 的范围内.数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数 ...

  6. Oracle不走索引的原因

    Oracle数据库操作中,为什么有时一个表的某个字段明明有索引,当观察一些语的执行计划确不走索引呢?如何解决呢?本文我们主要就介绍这部分内容,接下来就让我们一起来了解一下 . 不走索引大体有以下几个原 ...

  7. 【Azure 容器应用】在中国区Azure上创建的容器服务默认应用域名不全

    问题描述 在中国区Azure上,创建Container App服务,发现默认的应用程序URL只有前半段,并不是一个完整的域名.这是什么情况呢? 正常的Container App的URL格式为:< ...

  8. [学习笔记].Net5项目打包到Linux系统服务时遇到的坑

    ​如果按照官方文档的步骤手动安装.Net5 会有一个坑: 在 Linux 上手动安装 .NET - .NET | Microsoft Docs 在使用systemd打包.Net5服务的时候,无法运行, ...

  9. centos解决 pip is configured with locations that require TLS/SSL, however the ssl module in Python is not available.

    一.下载 openssl 编译安装 openssl 官方下载地址:https://www.openssl.org/source/ wget https://github.com/openssl/ope ...

  10. 英语自定义标签 <i:juzi><i:zhuyu>John Smith</i:zhuyu></i:juzi> 主语谓语宾语

    效果 John Smith died in World War Two. John Smith killed three enemy soldiers. <style> i\:juzi { ...