【赵渝强老师】PostgreSQL中表的碎片

在PostgreSQL中删除行的时候,这些行只是被标记为"dead",而不是真正从物理存储上进行删除了,因而空间也没有真的被释放回收。在PostgreSQL中除非进行自动的auto vacuum或者是手动的vacuum,否则数据块所占用的物理空间不会被回收。因此在物理存储空间被回收之前,会导致存储空间中存在很多空洞。如果表结构中包含动态长度字段,那么这些空洞甚至可能不能被PostgreSQL重新用来存储新的行。因此,大量随机的DELETE操作,必然会在数据文件中造成不连续的空白空间。而当插入数据时,这些空白空间也不会被利用起来,于是造成了数据的存储位置不连续。物理存储顺序与逻辑上的排序顺序不同,这种就是数据碎片。

对于大量的UPDATE,也会产生文件碎片化 , PostgreSQL的最小逻辑存储分配单位是数据块(Block),其默认值是8K。因此大量的更新操作也可能导致数据块的分裂(Block Split),即:同一个字段的数据可能存储在不同的数据块中。频繁的数据块分裂,会使得数据的存储变得稀疏,并且被不规则的数据填充,所以最终数据会有碎片。

点击这里查看视频讲解:【赵渝强老师】PostgreSQL中表的碎片

下面通过具体的步骤来演示如何清理表的碎片。

(1)创建一张新的表,并往表中插入5000万条记录

sql 复制代码
scott=# create table testfragement(tid int,tname varchar(20));
scott=# insert into testfragement select n,'myname_'||n from generate_series(1,50000000) n;

(2)查看表testfragement占用的容量大小。

sql 复制代码
scott=# select pg_size_pretty(pg_relation_size('testfragement'));

# 输出的结果如下:
 pg_size_pretty 
----------------
 2488 MB
(1 row)

(3)删除表中所有数据。

sql 复制代码
scott=# delete from testfragement;

(4)再次查看表testfragement占用的容量大小。

sql 复制代码
scott=# select pg_size_pretty(pg_relation_size('testfragement'));

# 输出的结果如下:
 pg_size_pretty 
----------------
 2488 MB
(1 row)

# 从输出的结果可以看出,尽管删除了表中的数据,表所占用的空间依然没有释放。

(5)查看表的状态信息。

sql 复制代码
scott=# \x
scott=# select * from pg_stat_user_tables where relname = 'testfragement';

# 输出的结果如下:
-[ RECORD 1 ]-------+------------------------------
relid               | 16574
schemaname          | public
relname             | testfragement
seq_scan            | 1
seq_tup_read        | 50000000
idx_scan            | 
idx_tup_fetch       | 
n_tup_ins           | 50000000
n_tup_upd           | 0
n_tup_del           | 50000000
n_tup_hot_upd       | 0
n_live_tup          | 0
n_dead_tup          | 49999426
n_mod_since_analyze | 0
n_ins_since_vacuum  | 0
last_vacuum         | 
last_autovacuum     | 2023-04-28 09:14:26.066678+08
last_analyze        | 
last_autoanalyze    | 2023-04-28 09:14:46.677939+08
vacuum_count        | 0
autovacuum_count    | 1
analyze_count       | 0
autoanalyze_count   | 1

其中:
n_live_tup 的数量是当前表的数据量。
n_dead_tup 的数据量是未回收的空间。

# 从参数n_dead_tup输出结果上看,表testfragement仍然占用了很多 "空闲" 数据块,其空间没有被回收。

(6)手动进行一下碎片的清理。

sql 复制代码
scott=# vacuum testfragement;

# 此时会产生后台相应的进程:
[root@mydb ~]# ps -ef|grep VACUUM
postgres  6649  3540 .... postgres: postgres [local] VACUUM 

(7)再次查看一下表的状态

sql 复制代码
scott=# \x
scott=# select * from pg_stat_user_tables where relname = 'testfragement';

# 输出的信息如下:
-[ RECORD 1 ]-------+------------------------------
relid               | 16574
schemaname          | public
relname             | testfragement
seq_scan            | 1
seq_tup_read        | 50000000
idx_scan            | 
idx_tup_fetch       | 
n_tup_ins           | 50000000
n_tup_upd           | 0
n_tup_del           | 50000000
n_tup_hot_upd       | 0
n_live_tup          | 0
n_dead_tup          | 0
n_mod_since_analyze | 0
n_ins_since_vacuum  | 0
last_vacuum         | 2023-04-28 09:23:05.463206+08
last_autovacuum     | 2023-04-28 09:18:11.434888+08
last_analyze        | 
last_autoanalyze    | 2023-04-28 09:14:46.677939+08
vacuum_count        | 1
autovacuum_count    | 2
analyze_count       | 0
autoanalyze_count   | 1

# 参数n_live_tup和n_dead_tup都变成了0,这说明表所占用的空间已经被释放回收。

(8)重新查看表testfragement占用的容量大小。

sql 复制代码
scott=# select pg_size_pretty(pg_relation_size('testfragement'));

# 输出的结果如下:
 pg_size_pretty 
----------------
 0 bytes
(1 row)
相关推荐
用户6279947182621 分钟前
南大通用GBase 8c数据库权限管理场景实践
数据库
八月瓜科技1 分钟前
擎策·知海全球专利数据库 专业专利检索赋能 规避无效研发提效创新
大数据·数据库·人工智能·科技·aigc
2401_833197734 分钟前
更优雅的测试:Pytest框架入门
jvm·数据库·python
倔强的石头_11 分钟前
Oracle 迁移 TCO 深度拆解:从隐性运维成本陷阱到全栈工具链破局
数据库
2501_9249526918 分钟前
Python深度学习入门:TensorFlow 2.0/Keras实战
jvm·数据库·python
2401_8914821720 分钟前
Python多线程与多进程:如何选择?(GIL全局解释器锁详解)
jvm·数据库·python
Predestination王瀞潞21 分钟前
6.3.1 软件->W3C XPath 1.0 标准(W3C Recommendation):XPath(XML Path Language)查询语言
xml·数据库·oracle
2401_8512729921 分钟前
用Python实现自动化的Web测试(Selenium)
jvm·数据库·python
快乐柠檬不快乐27 分钟前
Java连接电科金仓数据库(KingbaseES)实战指南
java·开发语言·数据库
AllData公司负责人38 分钟前
AllData数据中台通过集成DolphinScheduler+Seatunnel实现SAP HANA数据库同步到Doris数据仓库
数据库·数据仓库·sap hana