【MYSQL】GET_LOCK使用方法简单解析

背景描述: 在同一进程中,如果开了多个线程操作同一数据,为保证数据准确性,我们可以使用lock来保证互斥,但是如果是多个进程操作同一数据,lock就不行了,此时我们可以使用 基于MYSQL实现的分布式锁来实现。

sql 复制代码
SELECT GET_LOCK(@lockName, @timeout); //lockName==锁,timeout==等待时间,如果超时,不再等待

简单来说 ,就是多个进程去修改数据库的某一个数据(操作同一数据库),此时数据库为了保证准确性,会加一个锁,确保一次只能一个进程的一个线程去访问,其他访问的需要等待,直到持有这个锁。
GET_LOCK实现逻辑(简化版):

(1)MySQL 5.7 以及更早版本中一个连接(代码中构建和数据库的连接)只能持有一个命名锁,如果再次GET_LOCK('xxx', ...),会 自动释放之前的锁,再尝试获取新的锁。之后版本可以管理多个锁。

(2)当在代码中调用SELECT GET_LOCK(@lockName, @timeout); 时,mysql首先会查询当前全局锁表(hash结构)里是否有这个lockname

(3)如果没有,立即插入并记录当前连接信息,返回1。

(4)如果有,当前连接进入等待队列(内部用条件变量 COND_WAIT 实现)。

(5)如果在 timeout 秒内锁被释放 → 插入自己,返回 1。

(6)如果超时还没等到 → 返回 0(注意:此时锁依然在别人手上,不会自动释放)。

(7)异常情况:如果连接断开/SQL 出错 → 返回 NULL;如果等待过程中当前 session 被 kill → 直接中断,返回 NULL。
mysql查看数据库版本:

sql 复制代码
SHOW VARIABLES LIKE 'version';

代码示例:

csharp 复制代码
using System;
using MySql.Data.MySqlClient;
using System.Threading;

class Program
{
    static void Main()
    {
        string connStr = "Server=localhost;Database=test;User ID=root;Password=123456;";

        using (var conn = new MySqlConnection(connStr))
        {
            conn.Open();

            string lockName = "MyApp_DistributedLock";

            // 1. 尝试获取锁(等待超时 10 秒)
            using (var cmd = new MySqlCommand("SELECT GET_LOCK(@lockName, @timeout);", conn))
            {
                cmd.Parameters.AddWithValue("@lockName", lockName);
                cmd.Parameters.AddWithValue("@timeout", 10);

                var result = Convert.ToInt32(cmd.ExecuteScalar());
                if (result == 1)
                {
                    Console.WriteLine("✅ 获取到锁,可以安全执行任务");

                    try
                    {
                        // 模拟需要保护的临界区操作
                        Thread.Sleep(5000);
                    }
                    finally
                    {
                        // 2. 释放锁
                        using (var releaseCmd = new MySqlCommand("SELECT RELEASE_LOCK(@lockName);", conn))
                        {
                            releaseCmd.Parameters.AddWithValue("@lockName", lockName);
                            var releaseResult = Convert.ToInt32(releaseCmd.ExecuteScalar());
                            Console.WriteLine(releaseResult == 1 ? "🔓 锁已释放" : "⚠️ 锁释放失败");
                        }
                    }
                }
                else
                {
                    Console.WriteLine("❌ 未能获取到锁(可能被其他进程占用)");
                }
            }
        }
    }
}
相关推荐
山沐与山17 小时前
【数据库】PostgreSQL架构与索引深度剖析
数据库·postgresql·架构
不穿格子的程序员18 小时前
Redis篇6——Redis深度剖析:从单机到集群,Redis高可用进化史
数据库·redis·集群·主从·高可用·哨兵
阿坤带你走近大数据18 小时前
什么是元数据管理?(附具体实施方案供参考)
数据库·金融
俊男无期18 小时前
超效率工作法
java·前端·数据库
2301_8234380218 小时前
【无标题】解析《采用非对称自玩实现强健多机器人群集的深度强化学习方法》
数据库·人工智能·算法
中国胖子风清扬18 小时前
SpringAI和 Langchain4j等 AI 框架之间的差异和开发经验
java·数据库·人工智能·spring boot·spring cloud·ai·langchain
计算机学姐18 小时前
基于php的摄影网站系统
开发语言·vue.js·后端·mysql·php·phpstorm
Elastic 中国社区官方博客18 小时前
Elasticsearch:你是说,用于混合搜索(hybrid search)
大数据·数据库·人工智能·elasticsearch·搜索引擎·ai·全文检索
计算机学姐18 小时前
基于php的旅游景点预约门票管理系统
开发语言·后端·mysql·php·phpstorm
czlczl2002092518 小时前
高并发下的 Token 存储策略: Redis 与 MySQL 的一致性
数据库·redis·mysql