工作日志之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集群或数据库实例。

相关推荐
睡不醒男孩0308231 小时前
第二篇:深入探索开源数据库高可用:构建基于CLup的PostgreSQL生产级高可用与读写分离架构
数据库·postgresql·开源·clup
Micro麦可乐3 小时前
Spring Boot 实战:从零设计一个短链系统(含完整代码与数据库设计)
数据库·spring boot·后端·哈希算法·雪花算法·短链系统
码农阿豪4 小时前
从零到一:Spring Boot快速接入金仓数据库实战
数据库·spring boot·后端
鼎讯信通4 小时前
风电光缆运维提质增效:G-4000A 光缆故障追踪仪破解风场巡检难题
运维·网络·数据库
三十..4 小时前
MySQL 从入门到高可用架构实战精要
运维·数据库·mysql
cfm_29145 小时前
Redis五大基本数据结构底层了解
数据结构·数据库·redis
真实的菜6 小时前
Redis 从入门到精通(十二):典型业务场景实战 —— 排行榜、限流器、秒杀系统、Session 共享
数据库·redis·python
你想考研啊6 小时前
mysql数据库导出导入
数据库·mysql·oracle
十年编程老舅7 小时前
Linux DRM:底层逻辑与实践架构
数据库·mysql
The Sheep 20237 小时前
Vue复习
linux·服务器·数据库