侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

PostgreSQL数据库事务实现方法分析

数据库  /  管理员 发布于 6年前   188

本文实例讲述了PostgreSQL数据库事务实现方法。分享给大家供大家参考,具体如下:

事务简介

  • 事务管理器:有限状态机
    • 日志管理器
      • CLOG:事务的执行结果
      • XLOG:undo/redo日志
    • 锁管理器:实现并发控制,读阶段采用MVCC,写阶段采用锁控制实现不同的隔离级别

事务是所有数据库系统的一个基本概念。 一次事务的要点就是它把多个步骤捆绑成了一个单一的,不成功则成仁的操作。 其它并发的事务是看不到在这些步骤之间的中间状态的,并且如果发生了一些问题, 导致该事务无法完成,那么所有这些步骤都完全不会影响数据库。PostgreSQL为每条事务创建一个postgre进程,并发执行事务。采用分层的机制执行事务,上层事务块和底层事务。上层事务块是用户眼中的事务,用于控制事务执行的状态;底层事务是事务中的每条语句,可以改变上层事务块的状态。

上层事务块

每个postgre进程只有一个事务块,上层事务块记录着本次事务执行过程中的各个状态。

typedef enum TBlockState{  /* not-in-transaction-block states */  TBLOCK_DEFAULT,       /* idle */  TBLOCK_STARTED,       /* 执行简单查询事务 */  /* transaction block states */  TBLOCK_BEGIN,        /* 遇见事务开始BEGIN */  TBLOCK_INPROGRESS,     /* 事务正在执行中 */  TBLOCK_PARALLEL_INPROGRESS, /* live transaction inside parallel worker */  TBLOCK_END,         /* 遇见事务结束COMMIT/END的时候设置 */  TBLOCK_ABORT,        /* 事务出错,等待ROLLBACK */  TBLOCK_ABORT_END,      /* 事务出错,收到ROLLBACK */  TBLOCK_ABORT_PENDING,    /* 事务处理中,接收到ROLLBACK */  TBLOCK_PREPARE,       /* 事务处理中,收到PREPARE(分布式事务) */  /* subtransaction states */  TBLOCK_SUBBEGIN,      /* starting a subtransaction */  TBLOCK_SUBINPROGRESS,    /* live subtransaction */  TBLOCK_SUBRELEASE,     /* RELEASE received */  TBLOCK_SUBCOMMIT,      /* COMMIT received while TBLOCK_SUBINPROGRESS */  TBLOCK_SUBABORT,      /* failed subxact, awaiting ROLLBACK */  TBLOCK_SUBABORT_END,    /* failed subxact, ROLLBACK received */  TBLOCK_SUBABORT_PENDING,  /* live subxact, ROLLBACK received */  TBLOCK_SUBRESTART,     /* live subxact, ROLLBACK TO received */  TBLOCK_SUBABORT_RESTART   /* failed subxact, ROLLBACK TO received */} TBlockState;

常见的事务块状态转换图

 

  • startTransactionCommand:事务块中每条语句执行前都会调用。
  • commitTransactionCommand:事务块中每条语句执行结束都会调用
  • abortCurrentTransaction:事务块中语句执行错误,在调用点调用
  • BeginTransactionBlock:遇见BEGIN命令调用,状态变为TBLOCK_BEGIN
  • EndTransactionBlock:遇见END调用,可能成功提交,也可能回滚
  • AbortTransactionBlock:遇见ABORT指令调用

底层事务

底层事务是需要执行的每条命令,负责处理资源和锁的获取和释放,信号的处理,日志记录等等

typedef enum TransState{  TRANS_DEFAULT,       /* idle */  TRANS_START,        /* transaction starting */  TRANS_INPROGRESS,      /* inside a valid transaction */  TRANS_COMMIT,        /* commit in progress */  TRANS_ABORT,        /* abort in progress */  TRANS_PREPARE        /* prepare in progress */} TransState;

主要有四个函数:

  • StartTransaction:由BEGIN的startTransactionCommand调用,调用结束后事务块状态为TBLOCK_STARTED
  • CommitTransaction:由END的commitTransactionCommand调用,提交事务
  • AbortTransaction和CleanupTransaction:释放资源,恢复默认状态

分布式事务

PostgreSQL提供了分布式事务中的,两阶段提交的接口

并发控制

PostgreSQL采用MVCC的方式进行并发控制,每个事务看到的是一段时间前的数据快照。同时,MVCC并不能够解决所有问题,所以也提供了行级和表级的锁。

标准的事务隔离级别有4个,而PostgreSQL只实现了读已提交和可串行化。

锁

PostgreSQL实现了8种锁(可怕)

 

太多了,就记住几个吧。

  • 行共享锁:select for update/for share
  • 表共享锁:select
  • 行排他锁:insert/update/delete
  • 表排他锁:drop

加锁的对象

  • 表
    • 表锁
    • 会话锁
    • 扩展锁:新增表空间
  • 页:对索引页面
  • 元组:
  • 事务:

死锁处理

 

  • postgresql检测出最后一个等待的杀掉,oracle是第一个等待的杀掉
  • 死锁检测算法(等待图)

MVCC

关键词:

  • 基于事务ID
  • 行级多版本
  • 无回滚段,行内存储
    • 一次UPDATE,产生记录两个版本
    • 两个版本都存在页面内部
typedef struct HeapTupleFields{  TransactionId t_xmin;    /* Insert,Update事务 */  TransactionId t_xmax;    /* Delete,Update,Row Locks事务ID */  union  {    CommandId  t_cid;   /* 操作ID */    TransactionId t_xvac;  /* old-style VACUUM FULL xact ID */  }      t_field3;} HeapTupleFields;

cmin:插入该元组的命令在插入事务中的命令标识(从0开始累加)
cmax:删除该元组的命令在插入事务中的命令标识(从0开始累加)
ctid:相当于rowid , <数据块ID,偏移量>
XID:事务ID
Xid_snapshot:当前系统中未提交的事务
CLOG:事务状态日志(已提交的日志)

隔离级别

  1. RC:读已提交
    1. 两个事务可以并发更新同一行
    2. 一个事务更新,一个事务删除同一行,删除操作会上锁
  2. RR:读未提交,其实是snapshot isolation,(冲突状态会回滚)
  3. 可串行化:serialize snapshot isolation,比标准可串行化要高,通过加内存中的意向锁实现,不允许预加锁的数据被其他事务变更

数据可见性判断

  1. 记录的头部XID信息比当前事务更早(rr和ssi有这个要求,read commited没有这个要求,读已经提交可以读未来的事务!!)
  2. 记录头部的XID信息不在当前的XID_snapshot中,(记录上的事务状态不是未提交的事务)
  3. 记录头部的XID信息在CLOG中代表已提交。
  • MVCC需要判断该行数据在这个事务中的有效性,可见性,可更新性(需要锁的帮助才能正确执行隔离级别)
  • 判断条件:若xmin等于当前事务ID,则包含所有xmax=0(未被删除)的元组。
    若与xmin相等的事务ID对应的事务已经被提交,则包含所有xmax=0或xmax为当前事务ID的元组。
  • 实现概要
    • 对读不用加锁,对写加锁(只阻塞写),事务结束对比是否冲突

多行数据需要过期版本回收

  1. 页面级:页面访问时回收
  2. 表级/系统级: autovacuum; vacuum

日志

  1. pg_log:数据库活动日志(也就是数据库的操作日志);
  2. pg_xlog:事务日志,记录事务的执行过程,redo日志
  3. pg_clog:事务状态日志(pg_clog是pg_xlog的辅助日志),记录事务的结果。

希望本文所述对大家PostgreSQL数据库程序设计有所帮助。


  • 上一条:
    PostgreSQL数据库中窗口函数的语法与使用
    下一条:
    Abp.NHibernate连接PostgreSQl数据库的方法
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 分库分表的目的、优缺点及具体实现方式介绍(0个评论)
    • DevDB - 在 VS 代码中直接访问数据库(0个评论)
    • 在ubuntu系统中实现mysql数据存储目录迁移流程步骤(0个评论)
    • 在mysql中使用存储过程批量新增测试数据流程步骤(0个评论)
    • php+mysql数据库批量根据条件快速更新、连表更新sql实现(0个评论)
    • 近期文章
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • Laravel 11.15版本发布 - Eloquent Builder中添加的泛型(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2017-06
    • 2017-08
    • 2017-09
    • 2017-10
    • 2017-11
    • 2018-01
    • 2018-05
    • 2018-10
    • 2018-11
    • 2020-02
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2020-07
    • 2020-08
    • 2020-09
    • 2021-02
    • 2021-04
    • 2021-07
    • 2021-08
    • 2021-11
    • 2021-12
    • 2022-02
    • 2022-03
    • 2022-05
    • 2022-06
    • 2022-07
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-07
    • 2023-08
    • 2023-10
    • 2023-11
    • 2023-12
    • 2024-01
    • 2024-03
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客