工作日志之postgresql实现分布式锁

在PostgreSQL中实现分布式锁,有几种主流方案,各有侧重。其中最经典高效的是使用咨询锁(Advisory Locks)。

方案 核心原理 优点 缺点 适用场景

  1. 基于咨询锁 利用PG内置的 pg_advisory_lock(key) 函数在服务器内存中加锁。 性能极高、无表结构依赖、会话/事务级锁灵活。 锁与连接绑定,连接断开会释放;需自行处理死锁。 高并发、短任务(如秒杀、定时任务调度)。
  2. 基于行级锁 使用 SELECT ... FOR UPDATE 锁定表中的一条特定记录。 利用事务机制,可靠自动释放;锁状态可查。 性能不如咨询锁;需要创建锁表;事务持有时间长影响并发。 需要与业务数据强关联或跨事务复杂锁的场景。
  3. 基于排他锁 使用 LOCK TABLE ... IN ACCESS EXCLUSIVE MODE 锁整张表。 实现最简单粗暴。 并发性能最差,阻塞所有其他操作。 极少使用,仅用于极端维护操作。
    ⚠️ 重要前提:所有方案都要求连接到同一个PostgreSQL数据库集群,适用于分布式服务共享一个数据库的场景。若你的服务连接的是完全独立的不同数据库,则需寻求其他方案(如基于Redis或ZooKeeper)。

方案一: 基于咨询锁实现分布式锁

这是PostgreSQL为应用程序锁提供的原生、高性能方案。关键在于选择会话级或事务级锁。

(1)会话级锁:pg_advisory_lock(key),连接断开才释放,适合跨事务的长时间锁定。

(2)事务级锁:pg_advisory_xact_lock(key),事务结束(提交或回滚)即释放,更常用。

java 复制代码
-- 事务1:获取锁(这里使用事务级锁,key可以是一个或多个整数,或一个字符串的哈希值)
BEGIN;
SELECT pg_advisory_xact_lock(123456);
-- 执行你的临界区业务逻辑...
-- COMMIT; 提交后锁自动释放

-- 事务2:尝试获取同一个锁(会阻塞等待,直到事务1释放)
BEGIN;
SELECT pg_advisory_xact_lock(123456); -- 这里会等待
-- 获取成功后执行...
COMMIT;

非阻塞尝试加锁

java 复制代码
-- 使用 pg_try_advisory_xact_lock(key),立即返回 true/false,不会阻塞
BEGIN;
SELECT pg_try_advisory_xact_lock(123456);
-- 如果返回 false,表示未获取到锁,可以执行其他降级逻辑
COMMIT;

方法二: 基于行级锁(唯一性约束)

创建表

sql 复制代码
CREATE TABLE distributed_lock (
    lock_key VARCHAR(100) PRIMARY KEY,
    locked_by TEXT,
    locked_at TIMESTAMPTZ DEFAULT NOW()
);

获取锁(利用唯一性约束)

sql 复制代码
BEGIN;
-- 尝试插入一条记录作为获取锁
INSERT INTO distributed_lock (lock_key, locked_by) VALUES ('order_operation', 'service_01')
ON CONFLICT (lock_key) DO UPDATE SET
    locked_by = EXCLUDED.locked_by,
    locked_at = NOW()
WHERE distributed_lock.locked_at < NOW() - INTERVAL '30 seconds'; -- 实现简单的超时释放
-- 检查是否更新成功(即获取到锁)
GET DIAGNOSTICS row_count = ROW_COUNT;
IF row_count = 0 THEN
    -- 未获取到锁
    ROLLBACK;
ELSE
    -- 成功获取锁,执行业务逻辑...
    COMMIT; -- 提交后,锁记录依然存在,但可通过删除记录或更新状态来释放
END IF;

释放锁

sql 复制代码
-- 释放锁:删除记录或更新字段标识
DELETE FROM distributed_lock WHERE lock_key = 'order_operation';

以上方案适合没有redis,zookeeper等典型用于实现分布式锁的系统,并且前提是多个实例共享同一个pg集群或数据库实例。

相关推荐
2301_781571421 天前
Golang格式化输出占位符都有什么_Golang fmt占位符教程【通俗】
jvm·数据库·python
养肥胖虎1 天前
RAG学习笔记(3):区分数据库检索与RAG的使用场景
数据库·ai·rag
_ku_ku_1 天前
数据库系统原理 · 数据库应用开发 · 自学总结
数据库
No8g攻城狮1 天前
【人大金仓】wsl2+ubuntu22.04安装人大金仓数据库V9
java·数据库·spring boot·非关系型数据库
山峰哥1 天前
SQL慢查询调优实战:从全表扫描到索引覆盖的完整复盘
前端·数据库·sql·性能优化
代码中介商1 天前
Redis入门:5大数据类型全解析
数据库·redis·缓存
渣渣盟1 天前
数据库设计范式详解(纯小白版)
数据库·oracle·软考·数据库工程师
Jackyzhe1 天前
从零学习Kafka:消费者组重平衡
分布式·学习·kafka
夜雪闻竹1 天前
Cursor 对话导入:解析 SQLite 里的宝藏
数据库·sqlite·ai编程
海南java第二人1 天前
ClickHouse 部署模式完全指南:从单机到分布式集群的生产级选型
分布式·clickhouse