PostgreSQL 的 VACUUM 与 VACUUM FULL 详解

PostgreSQL 的 VACUUM 与 VACUUM FULL 详解

一、基本概念对比

特性VACUUMVACUUM FULL
定义常规维护操作,清理死元组激进重组操作,完全重写表数据
锁级别不阻塞读写(共享锁)排他锁(阻塞所有操作)
空间回收只标记空间为可用,不返还OS空间返还操作系统
索引处理不重建索引完全重建所有索引
执行速度快(增量式)慢(全量重写)
对系统影响

二、工作机制详解

1. VACUUM 工作原理

  • 死元组清理:标记被删除或更新旧版本的数据为"可重用"
  • 事务ID处理:防止事务ID回卷(冻结旧事务ID)
  • 更新统计信息:为查询优化器提供最新数据分布信息
  • 不减少物理文件大小:只是内部空间重用

2. VACUUM FULL 工作原理

  1. 创建表的全新副本
  2. 只将有效数据写入新存储
  3. 删除原始表文件
  4. 将新文件重命名为原表名
  5. 完全重建所有索引(因为元组物理位置改变)

三、使用场景对比

适合使用 VACUUM 的场景

  • 常规数据库维护(建议配置autovacuum)
  • 高并发OLTP系统
  • 频繁更新的表(每天或每小时)
  • 只需要空间重用,不需OS空间回收

适合使用 VACUUM FULL 的场景

  • 表膨胀严重(>30%空间浪费)
  • 准备进行大版本升级前
  • 长期未维护的历史表
  • 需要彻底重组表物理结构时

四、性能影响分析

VACUUM 影响

-- 典型资源占用
CPU: 5-15% 
IO: 中等(取决于表大小)
锁: 不阻塞查询,可能与DDL冲突
持续时间: 几分钟到几小时(大表)

VACUUM FULL 影响

-- 典型资源占用
CPU: 30-70%
IO: 非常高(读写全表数据)
锁: 完全阻塞表访问
持续时间: 几小时到几天(特大表)

五、实际操作示例

基本语法

-- 普通VACUUM
VACUUM [VERBOSE] [ANALYZE] [table_name];
-- 示例
VACUUM VERBOSE ANALYZE orders;

-- VACUUM FULL
VACUUM FULL [VERBOSE] [table_name];
-- 示例
VACUUM FULL VERBOSE large_table;

监控命令

-- 查看表膨胀情况
SELECT
    schemaname || '.' || relname AS table_name,
    pg_size_pretty(pg_total_relation_size(relid)) AS total_size,
    pg_size_pretty(pg_relation_size(relid)) AS data_size,
    n_live_tup AS live_tuples,
    n_dead_tup AS dead_tuples,
    round((n_dead_tup::numeric / (n_live_tup + n_dead_tup) * 100), 2) AS dead_tuple_percent,
    last_vacuum,
    last_autovacuum
FROM pg_stat_user_tables
WHERE n_live_tup > 0
ORDER BY dead_tuple_percent DESC;

六、最佳实践建议

  1. 常规维护策略

    • 启用并合理配置autovacuum
    • 对大表设置更频繁的vacuum阈值
    ALTER TABLE large_table SET (
      autovacuum_vacuum_scale_factor = 0.01,
      autovacuum_vacuum_threshold = 1000
    );
    
  2. VACUUM FULL 替代方案

    • 使用pg_repack扩展(在线重组,不阻塞读写)
    -- 安装后使用
    pg_repack -d dbname -t table_name
    
    • 手动创建新表交换(需要更多步骤)
  3. 特殊场景处理

    • 对于只读表,可以禁用autovacuum
    ALTER TABLE historical_data SET (
      autovacuum_enabled = false
    );
    
    • 紧急空间回收时考虑在维护窗口使用VACUUM FULL
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值