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

相关推荐
百结2143 小时前
Mysql数据库操作
数据库·mysql·oracle
keep one's resolveY4 小时前
时区问题解决
数据库
Leinwin4 小时前
OpenClaw 多 Agent 协作框架的并发限制与企业化规避方案痛点直击
java·运维·数据库
qq_417695054 小时前
机器学习与人工智能
jvm·数据库·python
漫随流水4 小时前
旅游推荐系统(view.py)
前端·数据库·python·旅游
ego.iblacat4 小时前
MySQL 服务基础
数据库·mysql
Maverick066 小时前
Oracle Redo 日志操作手册
数据库·oracle
l1t6 小时前
DeepSeek总结的为 pg_dumpall 添加非文本输出格式
postgresql
攒了一袋星辰6 小时前
高并发强一致性顺序号生成系统 -- SequenceGenerator
java·数据库·mysql
W.D.小糊涂6 小时前
gpu服务器安装windows+ubuntu24.04双系统
c语言·开发语言·数据库