PostgreSQL的分区表功能可以将大表拆分为多个小分区,提升查询和维护效率,而触发器在分区表场景中既可以配合原生分区机制补充功能,也可以作为自定义分区路由的实现方式,不同使用方式对应不同的作用和价值。

PostgreSQL原生分区表的默认行为
PostgreSQL 10及以上版本提供了声明式分区功能,创建分区表后,插入数据时数据库会自动根据分区键的规则将数据路由到对应的分区,不需要额外配置触发器。比如我们创建一个按时间分区的订单表:
-- 创建主分区表
CREATE TABLE orders (
order_id INT,
order_time DATE,
amount DECIMAL(10,2)
) PARTITION BY RANGE (order_time);
-- 创建2024年1月的分区
CREATE TABLE orders_202401 PARTITION OF orders
FOR VALUES FROM ('2024-01-01') TO ('2024-02-01');
-- 创建2024年2月的分区
CREATE TABLE orders_202402 PARTITION OF orders
FOR VALUES FROM ('2024-02-01') TO ('2024-03-01');
向orders表插入2024年1月的数据时,数据库会自动将数据存入orders_202401分区,这个过程由内核原生实现,性能比触发器路由更高。
触发器在分区表中的常见作用
1. 补充数据校验逻辑
原生分区只做路由判断,不会做额外的业务校验,我们可以通过触发器在插入或者更新数据前校验合法性。比如限制订单金额不能为负数:
-- 创建校验函数
CREATE OR REPLACE FUNCTION check_order_amount()
RETURNS TRIGGER AS $$
BEGIN
IF NEW.amount < 0 THEN
RAISE EXCEPTION '订单金额不能为负数,当前值为%', NEW.amount;
END IF;
RETURN NEW;
END;
$$ LANGUAGE plpgsql;
-- 在主分区表上创建触发器
CREATE TRIGGER trigger_check_order_amount
BEFORE INSERT OR UPDATE ON orders
FOR EACH ROW EXECUTE FUNCTION check_order_amount();
2. 实现审计日志记录
如果需要记录分区表的数据变更历史,可以在分区表上创建审计触发器,将变更记录同步到审计表中:
-- 创建审计表
CREATE TABLE orders_audit (
audit_id SERIAL PRIMARY KEY,
order_id INT,
operation_type TEXT,
operation_time TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
old_data JSONB,
new_data JSONB
);
-- 创建审计函数
CREATE OR REPLACE FUNCTION record_order_audit()
RETURNS TRIGGER AS $$
BEGIN
IF TG_OP = 'INSERT' THEN
INSERT INTO orders_audit (order_id, operation_type, new_data)
VALUES (NEW.order_id, 'INSERT', to_jsonb(NEW));
ELSIF TG_OP = 'UPDATE' THEN
INSERT INTO orders_audit (order_id, operation_type, old_data, new_data)
VALUES (NEW.order_id, 'UPDATE', to_jsonb(OLD), to_jsonb(NEW));
ELSIF TG_OP = 'DELETE' THEN
INSERT INTO orders_audit (order_id, operation_type, old_data)
VALUES (OLD.order_id, 'DELETE', to_jsonb(OLD));
END IF;
RETURN NULL;
END;
$$ LANGUAGE plpgsql;
-- 在主分区表上创建审计触发器
CREATE TRIGGER trigger_order_audit
AFTER INSERT OR UPDATE OR DELETE ON orders
FOR EACH ROW EXECUTE FUNCTION record_order_audit();
3. 自定义分区路由(替代原生分区)
如果使用PostgreSQL 10之前的版本,或者需要更灵活的分区规则(比如按哈希值路由到自定义分区),可以用触发器实现分区路由。示例如下:
-- 创建主表(不分区)
CREATE TABLE user_logs (
log_id INT,
user_id INT,
log_content TEXT
);
-- 创建两个自定义分区
CREATE TABLE user_logs_0 (LIKE user_logs INCLUDING ALL);
CREATE TABLE user_logs_1 (LIKE user_logs INCLUDING ALL);
-- 创建路由函数
CREATE OR REPLACE FUNCTION route_user_logs()
RETURNS TRIGGER AS $$
BEGIN
IF NEW.user_id % 2 = 0 THEN
INSERT INTO user_logs_0 VALUES (NEW.*);
ELSE
INSERT INTO user_logs_1 VALUES (NEW.*);
END IF;
RETURN NULL;
END;
$$ LANGUAGE plpgsql;
-- 创建路由触发器
CREATE TRIGGER trigger_route_user_logs
BEFORE INSERT ON user_logs
FOR EACH ROW EXECUTE FUNCTION route_user_logs();
两种分区方式的对比
我们可以通过下表对比原生分区和触发器自定义分区的差异:
| 对比项 | 原生分区 | 触发器自定义分区 |
|---|---|---|
| 路由性能 | 高,内核原生实现 | 较低,需要执行触发器函数 |
| 分区规则灵活性 | 仅支持范围、列表、哈希三种规则 | 可自定义任意路由逻辑 |
| 维护成本 | 低,数据库自动管理分区 | 高,需要手动维护分区和触发器逻辑 |
| 适用版本 | PostgreSQL 10及以上 | 所有PostgreSQL版本 |
使用注意事项
- 如果使用原生分区表,尽量将业务相关的触发器创建在主表上,避免每个分区单独创建触发器,减少维护工作量。
- 触发器会增加数据操作的额外开销,如果是高频写入的分区表,需要评估触发器的性能影响,避免不必要的触发器逻辑。
- 自定义触发器实现分区路由时,需要手动处理分区的创建、删除逻辑,还要考虑分区不存在时的异常处理。
注意:PostgreSQL 11及以上版本支持分区表的行级触发器自动继承到所有分区,不需要手动为每个分区创建触发器,进一步降低了维护成本。
PostgreSQL触发器分区表分区触发机制修改时间:2026-06-16 00:18:33