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

相关推荐
初次攀爬者1 天前
ZooKeeper 实现分布式锁的两种方式
分布式·后端·zookeeper
爱可生开源社区1 天前
2026 年,优秀的 DBA 需要具备哪些素质?
数据库·人工智能·dba
随逸1772 天前
《从零搭建NestJS项目》
数据库·typescript
加号32 天前
windows系统下mysql多源数据库同步部署
数据库·windows·mysql
シ風箏2 天前
MySQL【部署 04】Docker部署 MySQL8.0.32 版本(网盘镜像及启动命令分享)
数据库·mysql·docker
李慕婉学姐2 天前
Springboot智慧社区系统设计与开发6n99s526(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
数据库·spring boot·后端
百锦再2 天前
Django实现接口token检测的实现方案
数据库·python·django·sqlite·flask·fastapi·pip
tryCbest2 天前
数据库SQL学习
数据库·sql
jnrjian2 天前
ORA-01017 查找机器名 用户名 以及library cache lock 参数含义
数据库·oracle
十月南城2 天前
数据湖技术对比——Iceberg、Hudi、Delta的表格格式与维护策略
大数据·数据库·数据仓库·hive·hadoop·spark