正确地使用Redis的SETNX实现锁机制

setNX,是set if not exists 的缩写,也就是只有不存在的时候才设置, 设置成功时返回 1 , 设置失败时返回 0 。可以利用它来实现锁的效果,但是很多人在使用的过程中都有一些问题没有考虑到。

例如某个查询数据库的接口因为请求量比较大所以加了缓存,并设定缓存过期后刷新。当并发量比较大并且缓存过期的瞬间,大量并发请求会直接查询数据库导致雪崩。如果使用锁机制来控制只有一个请求去更新缓存就能避免雪崩的问题。下面是很多人下意识想到的加锁方法

$rs = $redis->setNX($key, $value);
if ($rs) {
    //处理更新缓存逻辑
    // ......
    //删除锁
    $redis->del($key);
}

通过 setNX 获取锁,如果成功了则更新缓存然后删除锁。其实这里有一个严重的问题:如果更新缓存的时候因为某些原因意外退出了,那么这个锁就不会被删除而一直存在,以至于缓存再也得不到更新。为了解决这个问题有人可能会想到给锁设置一个过期时间,如下

$redis->multi();
$redis->setNX($key, $value);
$redis->expire($key, $ttl);
$redis->exec();

因为 setNX 不具备设置过期时间的功能,所以要借助 Expire 来设置,同时需要使用 Multi/Exec 来确保请求的原子性,以免 setNX 成功了 Expire 却失败了。这样还有问题:当多个请求到达时,虽然只有一个请求的 setNX 可以成功,但是任何一个请求的 Expire 却都可以成功,这就意味着即便获取不到锁也可以刷新过期时间,导致锁一直有效,还是解决不了上面的问题。显然 setNX 满足不了需求,Redis从 2.6.12 起,SET 涵盖了 SETEX 的功能, SET 本身又包含了设置过期时间的功能,所以使用 SET 就可以解决上面遇到的问题

$rs = $redis->set($key, $value, array('nx', 'ex' => $ttl));
if ($rs) {
    //处理更新缓存逻辑
    // ......
    //删除锁
    $redis->del($key);
}

到这一步其实还是有问题的,如果一个请求更新缓存的时间比锁的有效期还要长,导致在缓存更新过程中锁就失效了,此时另一个请求就会获取到锁,但前一个请求在缓存更新完毕的时候,直接删除锁的话就会出现误删其它请求创建的锁的情况。所以要避免这种问题,可以在创建锁的时候需要引入一个随机值并在删除锁的时候加以判断

$rs = $redis->set($key, $random, array('nx', 'ex' => $ttl));
if ($rs) {
     //处理更新缓存逻辑
    // ......
    //先判断随机数,是同一个则删除锁
    if ($redis->get($key) == $random) {
        $redis->del($key);
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要在 C 语言中使用 Redis 实现分布式锁,你可以利用 RedisSETNX 命令和过期时间来实现SETNX 命令用于设置一个键的值,但只有在该键不存在时才会设置成功。通过设置一个带有过期时间的键,可以实现分布式锁的超时机制。 下面是一个示例代码,演示如何使用 Redis 实现分布式锁: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <hiredis/hiredis.h> #define REDIS_HOST "localhost" #define REDIS_PORT 6379 int main() { // 连接到 Redis 服务器 redisContext* context = redisConnect(REDIS_HOST, REDIS_PORT); if (context == NULL || context->err) { if (context) { printf("连接 Redis 服务器出错: %s\n", context->errstr); redisFree(context); } else { printf("无法连接到 Redis 服务器\n"); } return -1; } // 定义锁的名称和超时时间 const char* lockKey = "my_lock"; int lockTimeout = 10; // 尝试获取分布式锁 const char* lockValue = "locked"; redisReply* reply = (redisReply*)redisCommand(context, "SET %s %s NX PX %d", lockKey, lockValue, lockTimeout); if (reply == NULL) { printf("执行 SET 命令出错\n"); redisFree(context); return -1; } if (strcmp(reply->str, "OK") == 0) { printf("获取到分布式锁\n"); // 在这里执行你的临界区代码 // 释放分布式锁 redisReply* delReply = (redisReply*)redisCommand(context, "DEL %s", lockKey); if (delReply == NULL) { printf("执行 DEL 命令出错\n"); } else { printf("释放分布式锁\n"); freeReplyObject(delReply); } } else { printf("获取分布式锁失败\n"); } freeReplyObject(reply); redisFree(context); return 0; } ``` 在上述示例代码中,我们首先使用 `redisConnect` 函数连接到 Redis 服务器。然后,我们定义了要使用的锁的名称和超时时间。 使用 `redisCommand` 函数执行 `SET` 命令,将锁的名称作为键,锁的值和超时时间作为参数。如果 `SET` 命令执行成功,则表示成功获取到分布式锁。 在获取到锁后,我们可以在获取到锁后的代码块中执行临界区代码。最后,我们使用 `redisCommand` 函数执行 `DEL` 命令来释放分布式锁。 请确保已经正确安装并配置了 hiredis 库,并在编译时链接到该库。 这只是一个简单的示例代码,实际中可能还需要处理异常、错误处理等情况。具体实现方式可能因所使用Redis 版本和 C 语言库而有所不同。 希望这能帮助到你!如有其他问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值