PostgreSQL11 | 事务处理与并发控制

PostgreSQL11 | 事务处理与并发控制

本文章代码已在pgsql11.22版本上运行且通过,展示页由pgAdmin8.4版本提供,本文章第一次采用md文档,效果比csdn官方富文本编辑器好用,以后的文章都将采用md文档

事务管理简介

事物是pgsql中的基本工作单元,是用户定义的一个数据库操作序列,这些操作要么全做,要么全不做,是最低的工作单位

含义

事务必须要有明确的开始和结束点,pgsql中的每一条数据操作语句(例如select)都是隐式事务的一部分,即使只有一条数据操作语句,系统也会把这条语句当作一个事务,要么执行所有语句,要么什么都不执行

在pgsql中,由事务管理器负责管理事务运行的模块,结构如下:

事务管理器
锁管理器
进程管理
日志管理器
CLOG管理器
XLOG管理器
  • 事务管理器是事务系统的中枢,通过接受的信息处理下一步的事务操作。
  • 锁管理器主要提供在事务的写阶段并发控制所需要的各种锁,从而保证事务的各种隔离级别
  • 日志管理器主要是记录事务执行的状态和数据的变化过程

写到日志中的事务一般有两种

写入
写入
针对数据的操作
日志
针对任务的操作

针对数据的操作:插入、修改和删除,这些操作的对象是大量的数据

针对任务的操作:例如创建索引

当取消这些事务操作时,系统会自动执行这些操作的反操作,以保证系统的一致性,系统会自动生成一个检查点机制,这个检查点周期性的检查事务日志,如果在事务日志中,事务全部完成,那么检查点事务日志中的事务才会提交给数据库,并在事务日志中做一个检查点提交标识;若事务没有全部完成,则不会将事务日志中的事务提交数据库,并在事务日志中做一个检查点未提交的标识。

事务的恢复及检查点保证了系统的完整和可恢复

事务的属性

事务作为单个逻辑工作单元执行的一系列操作,一个逻辑单元必须有四个属性

  • 原子性:事务必须是原子工作单元
  • 一致性:事务完成时,必须使所有的数据都保持一致状态
  • 隔离性:由并发事务所做的修改必须与任何其他并发事务所做的修改隔离
  • 持久性:事务完成后,它对于系统的影响是永久性的

事务块管理的常用语句

在pgsql中,一个事务是通过把sql命令用BEGINCOMMIT命令包围实现的,语法如下:

BEGIN;
sql语句1.....
COMMIT;

事务块是指包围在BEGINCOMMIT之间的语句,同时注意两个关键词后一定要写;符号

常用事务块管理语句的含义如下:

  1. START TRANSACTION此命令表示一个新的事物块
  2. BEGIN表示初始化一个事务块,此后的语句都将在一个事务块中执行,直到出现COMMITROLLBACK,此命令与START TRANSACTION等价
  3. COMMIT提交事务
  4. ROLLBACK事务失败时执行回滚操作
  5. SET TRANSACTION设置当前事务的特性,对后面的事务没有影响

事务的应用案例

1.先建表people,如果已经有同样的表请先清空表
create table people
(id INT,name VARCHAR(40),age INT,info VARCHAR(50)
);
2.向people表中插入两名学生记录
insert into people values (1001,'斯大林',23,'18781218');
insert into people values (1002,'丘吉尔',25,'18741130');
3.查表people

在这里插入图片描述

4.接下来试着用事务块来进行插入操作

这里注意,我们的事务块到最后使用了回滚操作撤销了所有的对数据的操作

begin;
insert into people values (1003,'切格瓦拉',22,'19280614');
insert into people values (1004,'戴高乐',29,'18901122');
insert into people values (1004,'罗斯福',31,'18820130');
rollback transaction;
commit;
5.再次查表people

在这里插入图片描述

表中没有变化是因为事务撤销了对表的插入操作

PostgreSQL的并发控制

pgsql支持多用户共享同一数据库,但是当多个用户同时对同一数据库进行修改的时候会产生并发问题,所以并发控制的目标就是保证可用且高效且维护数据的完整性

并发导致的问题

数据库中的数据并发操作经常发生,而对数据的并发操作会带来脏读、幻读、不可重复读取等问题

  • 脏读:例如,当事务1读取的数据是事务2尚未完成,在过程中所临时产生的,后续又被事务2回滚掉了,而这一数据又被事务1读到了,那么所读取到的数据就是“错误”的,得到的数据是不真实的,数据本身仍未完成操作就被读,这个读取就是脏读
  • 幻读:例如,当事务1进行更新或删除时,事务2同时间对事务1所操作的部分进行了新增,则对于事务1来说,对数据的更新或删除不完全,因为新增的数据不在事务1操作的选中范围内,当事务1再次查看事务执行的结果时会查到事务2新增的数据,对于事物1产生了幻读
  • 不可重复性读取:当事务1在连续两次读取同一条记录时,事务2同步进行了数据的修改,对于事务1的两次查询结果是不相同的

pgsql利用多版本控制(MVCC)来维护数据的一致性,当检索数据时,每个事务看到的都是一小段时间之前的数据快照,而不是当前数据库真实的状态,这样对每个数据库会话进行隔离,就可以避免一个事务因为其他事务的操作而导致不一样的数据

四个事务隔离级别

在pgsql中,定义了4个事务隔离级别

  • 读未提交
  • 读已提交
  • 可重复读
  • 可串行读

在pgsql中,可以请求四种可能的事务隔离级别中的一种,但在其内部实际上只有两种独立的隔离级别:读已提交可串行读

读已提交是默认的隔离级别,容易出现幻读不可重复性读取问题,当两个事务对同一条数据更新时,第一个事务运行时,第二个事务需要等第一个事务提交或回滚才能进行

可串行化是最严格的事务隔离,不会出现上述三种问题,当两个事务对同一条数据更新时,串行化事务会等待第一个事务的提交或回滚,当第一事务提交了则回滚串行化事务,从头开始整个事务,如果第一个事务回滚了则忽略其影响,保证了串行化事务执行的时候是基于第一个事务完成后的实际数据进行操作

锁机制

MVCC并不能解决所有的并发控制情况,所以必须引入锁机制来保证事务的并发

锁的类型

  • SpinLock(自旋锁):使用互斥信号,与操作系统和硬件环境联系比较密切,封锁时间短,没有等待队列和死锁检测机制,事务结束后不能自动释放

  • LWLock(轻量级锁):主要提供对共享存储器的数据结构的互斥访问,特点主要是有等待队列和无死锁检测,事务结束后自动释放

    LWLock锁分为排他模式和共享模式,排他模式用于增删改,共享模式用于查

  • RegularLock(常规锁):一般数据库事务管理中所指的锁,有等待队列、死锁检测和自动释放的特点

    常规锁支持的模式有8种,按排他级别从低到高分别为:

    1. ACCESS SHARE(访问共享锁):查询命令将会在查询的表上获取访问共享锁
    2. ROW SHARE(行共享锁):使用SELECT FOR UPDATE(锁定查询结果中的每一行,而不是整个表)和SELECT FOR SHARE(对查询结果中的每一行施加共享锁。这意味着其他事务仍然可以读取这些行,但不能修改或删除它们,直到当前事务提交或回滚)命令时会获得行共享锁
    3. ROW EXCLUSIVE(行排他锁):使用UPDATEDELETEINSERT命令会在目标表上获得行排他锁
    4. SHARE UPDATE EXCLUSIVE(共享更新排他锁):使用VACUUM(当表中的行被更新或删除时,PostgreSQL不会立即从物理存储中移除这些行,而是将它们标记为不再需要(即死元组)。VACUUM命令会扫描表并移除这些死元组,使空间能够被重新利用)、ANALYZE(用于收集数据库中表和索引的统计信息,如列的数据分布、唯一值数量、空值数量等)、CREATE INDEX CONCURRENTLY(允许你在不锁定表的情况下创建索引。这意味着在创建索引的过程中,其他用户仍然可以对表进行读取和写入操作,而不会受到创建索引操作的影响)语句时使用共享排他锁
    5. SHARE(共享锁):使用CREATE INDEX语句请求时使用共享锁
    6. SHARE ROW EXCLUSIVE(共享行排他锁):和排他锁类似,只是允许共享
    7. EXCLUSIVE(排他锁):阻塞行共享和SELECT FOR UPDATE时使用排他锁
    8. ACCESS EXCLUSIVE(访问排他锁):如果没有声明指定模式,此模式为默认模式

死锁

多用户环境下,死锁的发生是由两个事务都锁定了不同的资源,又同时都在申请对方锁定的资源,互不释放,互相申请,且不更改条件,持续申请进而处于一种永久等待的状态

死锁的四个必要条件
  • 请求与保持条件
  • 非剥夺条件
  • 循环等待条件
  • 互斥条件
减少死锁的策略
  1. 在所有事务中以相同的次序使用资源
  2. 使事务尽可能简短并在同一批处理中
  3. 为死锁超时参数设置一个合理的范围,一般为3~30分钟,若超时则放弃本次操作,避免进程挂起
  4. 避免在事务内和用户进行交互,减少资源的锁定时间
  5. 使用较低的隔离级别,相比较高的隔离级别能够有效减少持有的共享锁的时间,减少锁之间的竞争

锁的语法

在pgsql中锁定一个表,语法如下:

LOCK [TABLE] name[,name2,..][IN lockmode MODE][NOWAIT]

作者的话(Alvin):

本文根据原书《PostgreSql11 从入门到精通》(清华大学出版社)第12章总结整理,为提问与解答可以帮助更多人,本博客模拟GitHub的issue方案,所以私信已关,有问题请在评论区直接指正与提问,允许转发、复制或引用本文章,必须遵守开源法则注释来源与作者,感谢您的阅读

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/408893.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

跨越语言障碍:2024四款实用在线翻译工具评测!

随着技术的不断进步,各种在线翻译工具层出不穷,帮助我们轻松跨越语言的鸿沟。今天,让我们一起探索几款非常实用的在线翻译工具! 福昕在线翻译 直达链接: https://fanyi.pdf365.cn/ 福昕在线翻译是一款专注于文档翻…

LeetCode 热题 100 回顾

目录 一、哈希部分 1.两数之和 (简单) 2.字母异位词分组 (中等) 3.最长连续序列 (中等) 二、双指针部分 4.移动零 (简单) 5.盛最多水的容器 (中等) 6…

jmeter中导入java方法并使用

1、首先打开idea,在idea中点击File-New-Project ,创建一个项目,项目名为JmtOne 2、项目创建完成后,直接在main函数中定义两个方法,记住该主函数的包名跟类名,后面会用到,在类中定义了两种拆分字符串的方法&…

iLogtail 开源两周年:感恩遇见,畅想未来

早在上世纪 60 年代,早期的计算机(例如 ENIAC 和 IBM 的大型机)在操作过程中会输出一些基本的状态信息和错误报告,这些记录通常通过打印机输出到纸带或纸卡上,用于跟踪操作流程和调试,最早期的日志系统借此…

Matlab基本知识

🌈个人主页:羽晨同学 💫个人格言:“成为自己未来的主人~” %% Matlab基本的小常识 % (1)在每一行的语句后面加上分号(一定要是英文的) a3; a5; % (2)多行注释:选中要注释的若干语句,快捷键CtrlR % a3; %…

MinIO 企业级人工智能存储的数据和驱动器同等重要

为什么会这样?这是因为硬件故障发生在不同的级别。有一些中断会导致整个站点瘫痪。然后,会出现中断,导致集群中的一部分节点瘫痪。但是,在更精细的硬盘驱动器位级别也存在故障,这些故障需要复制本身无法提供的另一种类…

android studio 新建java工程, 安卓新建项目,android studio2024 如何新建java项目

主要解决,新增安卓工程,没有java选项 1. 点击左上角FIle -> New -> 2. 选择 no activity 选项, 然后next 3. langua 就可以选择java 了。name自己定义项目名称,项目存储地址,包名。 配置完成选择finish. 4. fin…

MyBatis XML配置文件(下)

MyBatis的开发有两种方式:1、注解 2、XML。使用MyBatis的注解方式,主要是来完成一些简单的增删改查功能。如果需要实现复杂的SQL功能,建议使用XML来配置映射语句,也就是将SQL语句写在XML配置文件中。 MyBatis XML开发的方式需要以…

面向对象02:构造器详解

本节内容视频链接:面向对象05:构造器详解_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV12J41137hu?p64&vd_sourceb5775c3a4ea16a5306db9c7c1c1486b5 构造器(‌Constructor)‌是一种特殊的方法,‌用于初始…

linux下的oracle启动命令

一、服务器断电后,手工启动oracle数据库步骤如下: 1、进入数据库服务器,切换到oracle用户,命令:su - oracle 2、启动数据库,命令: 1) sqlplus / as sysdba 2) startup 3)如果数据库已…

【Qt笔记】Qt界面显示时间

目录 一、前言 二、基本概念 三、代码实现 1. 获取本地时间,并将其转换成自己想要的格式 2.创建一个QLabel控件用于显示时间字符串 3.创建QTimer定时器更新时间 四、优化 1. 格式优化 1.1 初始化 1.2 获取星期 1.3 更改格式 1.4 定时器超时函数 1.5 …

代码随想录 | day 15 | 二叉树part03

完全二叉树的节点个数 方法一:可以用递归法遍历一遍左子树和右子树的个数之和再加1等于全部节点个数 class Solution { public:int getcount(TreeNode* cur){if(curNULL) return 0;int leftcount getcount(cur->left);int rightcount getcount(cur->right…

以简单的例子从头开始建spring boot web多模块项目(四)-多模块工具类

目的是为了验证主工程调用工具工程。 1、新建模块&#xff0c;名称为WebTool 同样为Maven Archetype&#xff0c;类型为 org.apache.maven.archetypes:maven-archetype-quickstart 2、修改pom.xml 增加spring-boot-starter的依赖。 <dependency><groupId>org.spri…

【科研绘图】【分条热力图】:附Origin详细画图流程 + 案例分析

目录 No.1 理解分条热力图 No.2 画图流程 1 导入数据&#xff0c;绘制图形 2 设置绘图细节 3 色阶控制 4 设置坐标轴 5 效果图 No.3 案例分析 1 案例一 2 案例二 No.1 理解分条热力图 分条热力图&#xff0c;基于数据映射和颜色编码&#xff0c;是在热力图的基础上进…

【Hot100】LeetCode—437. 路径总和 III

目录 1- 思路前缀和哈希表dfs 2- 实现⭐437. 路径总和 III——题解思路 3- ACM 实现 题目连接&#xff1a;437. 路径总和 III 1- 思路 前缀和哈希表dfs ① 前缀和 求二叉树的前缀和&#xff0c;每求一次用一个 sum 传参记录更新 ② 哈希表 key 为前缀和 &#xff0c;value…

RISCV汇编编程讲解

第一章 引言 为什么要讲riscv&#xff1f; riscv的特点&#xff1a; -诞生于顶尖学术机构&#xff1a;诞生于加州大学伯克利分校的体系结构研究院。吸引了大批的顶尖企业参与&#xff08;e.g. 谷歌、华为、高通、阿里巴巴为rsicv的发展提供了大量的资金支持和贡献了技术和人才…

Oracle Linux 7.9 安装minikube体验

1.环境信息 前置所需&#xff1a; 操作系统&#xff1a;Oracle Linux 7.9 虚拟机配置&#xff1a;CPU:4核 内存&#xff1a;4G 容器&#xff1a;docker 26.1.4 安装minikube后环境&#xff1a; minikube: v1.33.1 kubernetes:v1.23.3 minukube体验说明&#xff1a;使用Virtua…

flume--数据从kafka到hdfs发生错误

解决&#xff1a; #1.将flume自带的依赖删除 mv /opt/installs/flume1.9/lib/guava-11.0.2.jar /opt/installs/flume1.9/lib/guava-11.0.2.jar.bak #2.将hadoop的依赖发送到flume下 cp /opt/installs/hadoop3.1.4/share/hadoop/common/lib/guava-27.0-jre.jar /opt/installs/f…

【C++ Primer Plus习题】5.9

问题: 解答: #include <iostream> #include <cstring> using namespace std;#define SIZE 20int main() {string words[SIZE];string done "done";int count 0;while (true){cout << "请输入单词:" << endl;cin >> words…

中国发布首个AI集成Linux开源操作系统

B站&#xff1a;啥都会一点的研究生公众号&#xff1a;啥都会一点的研究生 AI圈最近又发生了啥新鲜事&#xff1f; 中国大模型市场迎来新格局&#xff1a;百度、商汤、智谱位列前三 国际数据公司&#xff08;IDC&#xff09;于首次发布了《中国大模型平台市场份额&#xff0…