PostgreSQL系统隐藏字段

概述

在PostgreSQL数据库中,每个表都会有几个系统字段,这些字段是由系统隐含定义的。正常情况下是不对用户进行显示的。但是也可以查看到。通过了解这些隐藏字段。对pg的体系架构了解更深一步

oid

对象标识符,是各种系统表的主键。是一个四字节的无符号的整数
系统不会给用户创建的表增加一个oid字段,但是可以使用‘with oids’选项为表增加oid字段。另外,表中的oid生成的序列值为全局的。并不是从1开始的。如下

postgres=# create table test03 (id int) with oids;
CREATE TABLE
postgres=# insert into test03 values (1);
INSERT 16458 1
postgres=# insert into test03 values (2);
INSERT 16459 1

postgres=# select oid,id from test03 ;
  oid | id 
-------+----
 16458 | 1
 16459 | 2
(2 rows)

数据库,表,索引,视图等,都有一个oid标志符,

数据库
postgres=# select oid,datname from pg_database;
  oid | datname  
-------+-----------
 13878 | postgres
 16384 | stephen
     1 | template1
 13877 | template0

表或索引
postgres=# select oid,relname from pg_class where relname ~ 'test';
  oid | relname   
-------+-------------
 16398 | test01
 16447 | test02
 16453 | test02_pkey
 16455 | test03
 16401 | test_view

postgresql 的数据文件存储,就是通过oid进行的,在默认表空间下,存储的数据目录,就是对应的数据库的oid

[root@stephen pgdata]# ls base/
1 13877 13878 16384
[root@stephen pgdata]# ls base/16384/
112 13723_vm 16411 2601 2610 2620_vm 2679 2756 3079_fsm 3440 3598_vm 3997

xtid

ctid表示数据行的在它所处的表内的物理位置。ctid字段的类型是是tid。尽管tid可以非常快的定位数据行。但是每次vacuum full之后。数据行在块内的物理位置会移动。即ctid会发生变化。所以ctid不能长期作为行标识符

postgres=# select ctid,* from test02;
 ctid | id | name  
-------+----+-------
 (0,1) | 1 | dasda
 (0,3) | 2 | ceshi
(2 rows)

由上面可以看到,ctid由两个数字组成,第一个数字表示物理块号,第二个数字表示在物理块中的行数

xmin、xmax、cmin、cmax

这四个字段在mvcc中实现的中用于控制数据行是否对用户可见,PG会将修改前后的数据都存储在相同的结构中,分以下几种情况

  • 新插入一行时:将新插入的行的xmin填写为当前事务的事务ID,xmax填0
  • 修改某一行时:实际上是先插入一行,旧的行上的xmin不变,旧行上的xmax改为当前事务的id,新的行上的xmin填写为当前事务ID,xmax填写为0
  • 删除一行时:把被删除行上的xmax填写当前事务的ID

换句话说,xmin就是标记插入数据行的事务ID,而xmax就是标记删除数据行的事务ID

postgres=# select xmin,xmax,cmin,cmax,id,name from test02;
 xmin | xmax | cmin | cmax | id | name   
------+------+------+------+----+----------
  648 | 0 | 0 | 0 | 1 | dasda
  649 | 0 | 0 | 0 | 2 | dasdasda
(2 rows)
postgres=# update test02 set name = 'ceshi' where id =2;
UPDATE 1
postgres=# select xmin,xmax,cmin,cmax,id,name from test02;
 xmin | xmax | cmin | cmax | id | name  
------+------+------+------+----+-------
  648 | 0 | 0 | 0 | 1 | dasda
  651 | 0 | 0 | 0 | 2 | ceshi
(2 rows)

cmin和cmax用于判断同一个事务内的不同命令导致行版本变化的是否可见,如果一个事务内的所有命令都是严格按照顺序执行的,那么每个命令都能看到之前该事务内的所有变更。但是在一般的编程中数组或列表遍历时,就有可能会导致出现逻辑错误。所以需要这个隐藏字段来控制

上一篇:EnterpriseDB(阿里云PPAS)自建与使用


下一篇:PostgreSQL的设置开机自启动