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