当前位置: 首页 > news >正文

PostgreSQL中所的锁

       为了确保复杂的事务可以安全地同时运行,PostgreSQL提供了各种级别的锁来控制对各种数据对象的并发访问,使得对数据库关键部分的更改序列化。事务并发运行,直到它们尝试获取互相冲突的锁为止(比如两个事务更新同一行时)。当多个事务同时在数据库中运行时,并发控制是一种用于维持一致性和隔离性的技术,在PostgreSQL中,使用快照隔离Sanpshot Isolation (简称SI) 来实现多版本并发控制,同时以两阶段锁定 (2PL) 机制为辅。在执行DDL时使用2PL,在执行DML时使用SI

         在PostgreSQL中,最主要的是表级锁与行级锁,此外还有页锁、咨询锁

常用概念

系统锁与事务锁

原子操作:PostgreSQL为了支持无锁编程,提供了系列的原则操作,包括内存屏障,CAS(Compare And Swap),TAS(Test And Set)等

自旋锁(Spin lock):是一种和硬件结合的互斥锁,它借用了硬件提供的原子操作原语来对一些共享变量进行封锁,通常适用于临界区比较小的情况

轻量锁(Lightweight lock):PostgreSQL中进程需要对共享内存进行频繁的读写操作,轻量锁主要是保护这些共享内存中的数据结构。它是一种读写锁,有共享排它两种模式

常规锁(Regular lock):对数据库对象加锁,PostgreSQL两阶段锁就是借助常规锁实现的。根据封锁对象的不同,他有分成了不同粒度,如对表、页面、元祖、事务ID等分别加锁。已最常见的表锁为例,当不同的事务操作一个表时,会尝试通过表的Oid来构造LockTag,这样每个数据库对象都会有一个唯一标识,然后根据这个唯一的标识到锁表中申请锁。postgreSQL数据库将常规锁分层了8个不同的等级,不同的操作需要使用不同等级的常规锁。

常规锁的级别

锁模式说明
AccessShareLock(1)当对一个对象进行查询(select)操作,会申请该类型的锁,该锁是最低级别的锁,相当于读写锁中的共享锁
RowShareLock(2)当查询指定FOR UPDATE/SHARE时,会申请该类型的锁
RowExclusiveLock(3)当对数据对象做增删改操作是,会申请该类型的锁,INSERT、DELETE、UPDATE
ShareUpdateExclusiveLock(4)VACUUM(non-FULL)、ANALYZE、CREATE INDEX CONCURRENTLY, ALTER TABLE
ShareLock(5)主要用于创建索引时申请该类型的锁, CREATE INDEX
ShareRowExclusiveLock(6)和ExclusiveLock相似,但和RowShareLock兼容。
CREATE TRIGGER, ALTER TABLE
ExclusiveLock(7)和AccessExClusiveLock类似,但和最低级别的读锁AccessShareLock兼容
AccessExclusiveLock(8)在对元数据(系统表)做DDL操作时,会申请该类型的锁,AccessExclusiveLock与其他所有的锁模式都不相容
DROP,TRUNCATE,VACUUM FULL, LOCK TABLE

两阶段锁

对象锁

对象锁是在共享内存中的,受到两个参数值的限制:max_locks_per_transaction×max_connections

对象锁可以通过pg_locks这个视图来查询

 

SELECT 
l.pid,
a.datname AS database,
c.relname AS table,
l.mode AS lock_mode,
l.granted AS granted,
a.usename AS username,
a.query AS query
FROM 
pg_locks l
JOIN 
pg_stat_activity a ON l.pid = a.pid
JOIN 
pg_class c ON l.relation = c.oid

       如果资源已经锁定在不兼容的模式中,那么试图获取锁的事务将排队等待,直到释放锁。等待事务不消耗处理器资源:涉及的后端进程«休眠»,当资源空闲时被操作系统唤醒

对象类型

  • relation
  • transactionid/virtualxid
  • tuple
  • extend
  • object
  • page
  • advisory

对象级别的锁

行锁

行锁是如何实现的?

问题

HeapTupleSatisfiesMVCC中的HEAP_IS_LOCKED是干什么的?

为什么HEAP_IS_LOCKED就返回true?

调用栈

  • HEAP_XMAX_SHARED_LOCK
  • HEAP_XMAX_EXCL_LOCK
  • heap_lock_tuple
  • ExecLockRows

heap_lock_tuple的逻辑

  1. HeapTupleSatisfiesUpdate:
    HeapTupleBeingUpdated:已插入,正在被修改,还没有提交。
    多个事务在lock元组,而且至少有一个在运行
    仅仅有一个事务在修改元组,但还没有提交
  2. 拿锁,修改pg_multixact, 修改flag, 放锁

tuple锁

PostgreSQL中的行锁

转载文章:PostgreSQL中所的锁 - 知乎

相关文章:

  • MQTT通信协议使用说明
  • 防雷接地+防雷工程施工综合方案
  • 最新红盟云卡个人自动发卡开源系统源码+全开源无加密+虚拟商品在线售卖平台
  • 4、FFmpeg命令行操作5
  • 无人驾驶相关硬件汇总
  • 如何将 Docsify 项目部署到 CentOS 系统的 Nginx 中
  • echarts 多toolti同时触发图表实现
  • 数理统计的基本概念(二)
  • 操作系统安装在哪里?
  • Spark 平障录
  • YOLO目标检测——无人机检测数据集下载分享【含对应voc、coco和yolo三种格式标签】
  • Android studio Build Log乱码+错误: 找不到符号符号
  • QT在线安装指南
  • Mysql查看Binlog文件
  • 手机运行内存大揭秘:探索你手机的超级大脑!
  • 自己简单写的 事件订阅机制
  • E-HPC支持多队列管理和自动伸缩
  • Idea+maven+scala构建包并在spark on yarn 运行
  • IP路由与转发
  • JavaScript服务器推送技术之 WebSocket
  • JavaScript函数式编程(一)
  • Java程序员幽默爆笑锦集
  • java正则表式的使用
  • orm2 中文文档 3.1 模型属性
  • Spark in action on Kubernetes - Playground搭建与架构浅析
  • SpringCloud(第 039 篇)链接Mysql数据库,通过JpaRepository编写数据库访问
  • 前端技术周刊 2018-12-10:前端自动化测试
  • 浅谈Kotlin实战篇之自定义View图片圆角简单应用(一)
  • 如何胜任知名企业的商业数据分析师?
  • 如何使用 OAuth 2.0 将 LinkedIn 集成入 iOS 应用
  • 如何使用Mybatis第三方插件--PageHelper实现分页操作
  • 如何在GitHub上创建个人博客
  • 如何正确配置 Ubuntu 14.04 服务器?
  • 使用agvtool更改app version/build
  • 学习JavaScript数据结构与算法 — 树
  • 译自由幺半群
  • 终端用户监控:真实用户监控还是模拟监控?
  • # Python csv、xlsx、json、二进制(MP3) 文件读写基本使用
  • #android不同版本废弃api,新api。
  • #pragma预处理命令
  • #WEB前端(HTML属性)
  • (2015)JS ES6 必知的十个 特性
  • (27)4.8 习题课
  • (4.10~4.16)
  • (附源码)计算机毕业设计ssm基于B_S的汽车售后服务管理系统
  • (四)搭建容器云管理平台笔记—安装ETCD(不使用证书)
  • (一)Mocha源码阅读: 项目结构及命令行启动
  • (一)Thymeleaf用法——Thymeleaf简介
  • .net core 源码_ASP.NET Core之Identity源码学习
  • .NET Core日志内容详解,详解不同日志级别的区别和有关日志记录的实用工具和第三方库详解与示例
  • .NET Project Open Day(2011.11.13)
  • .net(C#)中String.Format如何使用
  • .NET/C# 使窗口永不激活(No Activate 永不获得焦点)
  • /boot 内存空间不够
  • @DateTimeFormat 和 @JsonFormat 注解详解