1、设置锁定的过期时间:当前的 Unix 时间戳 + Redis锁定超时时间,单位为秒(3),编辑文件:/common/config/params.php,如图1
'lock' => [ 'keyPrefix' => 'lock:', //Redis锁定 key 前缀 'timeOut' => 3, //Redis锁定超时时间,单位为秒 ],
2、获取相关的设置参数,编辑文件:/api/models/redis/GameCategory.php,如图2
// 设置锁定的过期时间 $time = time(); $lockKey = Yii::$app->params['lock']['keyPrefix'] . 'game_category'; $lockExpire = $time + Yii::$app->params['lock']['timeOut'];
3、获取 Redis 连接,以执行相关命令,编辑文件:/api/models/redis/GameCategory.php,如图3
// 获取 Redis 连接,以执行相关命令 $redis = Yii::$app->redis;
4、获取锁定,如图4
// 获取锁定 $executeCommandResult = $redis->setnx($lockKey, $lockExpire);
注:SETNX key value
将key设置值为value,如果key不存在,这种情况下等同SET命令。 当key存在时,什么也不做。SETNX是”SET if Not eXists”的简写。
5、返回0,表示已经被其他客户端锁定,如图5、6、7
// 返回0,表示已经被其他客户端锁定 if ($executeCommandResult == 0) { // $fileName = microtime(true); // file_put_contents('./../runtime/0-' . $fileName . '.txt', '1'); // 防止死锁,获取当前锁的过期时间 $lockCurrentExpire = $redis->get($lockKey); // $fileName = microtime(true); // file_put_contents('./../runtime/6-' . $fileName . $lockCurrentExpire . '.txt', '1'); // 判断锁是否过期,如果已经过期 if ($time > $lockCurrentExpire) { // $fileName = microtime(true); // file_put_contents('./../runtime/1-' . $fileName . '.txt', '1'); // 释放锁定 // $redis->del($lockKey); // 获取锁定 // $executeCommandResult = $redis->setnx($lockKey, $lockExpire); // 防止并发锁定,检查存储在 key 的旧值是否仍然是过期的时间戳,如果是,则获取锁定,否则返回假 $executeCommandResult = $redis->getset($lockKey, $lockExpire); if ($lockCurrentExpire != $executeCommandResult) { // $fileName = microtime(true); // file_put_contents('./../runtime/2-' . $fileName . '.txt', '1'); return ['status' => false, 'code' => 0, 'message' => '']; } // $fileName = microtime(true); // file_put_contents('./../runtime/3-' . $fileName . '.txt', '1'); } // 返回0,表示已经被其他客户端锁定,且不存在死锁,返回假 if ($executeCommandResult == 0) { // $fileName = microtime(true); // file_put_contents('./../runtime/4-' . $fileName . '.txt', '1'); return ['status' => false, 'code' => 0, 'message' => '']; } } // $fileName = microtime(true); // file_put_contents('./../runtime/5-' . $fileName . '.txt', '1');
6、释放锁定,如图8
注:DEL key [key …]
如果删除的key不存在,则直接忽略。
7、对于在第5点,判断锁是否过期,如果已经过期,注释掉释放锁定与获取锁定,为了防止并发锁定,可做以下测试流程以验证
8、先注释掉释放锁定,以模拟:如果客户端失败,崩溃或者无法释放锁,会发生什么?的问题,如图9
9、在判断锁是否过期,如果已经过期,这处代码段中,采用第一种算法,且将file_put_contents全部取消注释,如图10
10、在Redis中,执行命令:FLUSHDB,清空所有key,如图11
11、执行并发请求测试,设置线程数为10,如图12、13
12、查看/api/runtime目录下所生成的文件,以0、4、6开头的文件数量皆为9,以5开头的文件数量为1,正常,如图14
13、在Redis中,删除除了lock:game_category外的所有业务相关key,即以game_category开头的key,以模拟:锁定已经过期,如图15
14、删除/api/runtime目录下所生成的文件,如图16
15、执行并发请求测试,设置线程数为10,如图17、18
16、查看/api/runtime目录下所生成的文件,以0、6开头的文件数量为10,以1、4开头的文件数量皆为8,以5开头的文件数量为2,如图19
注1:以5开头的文件数量为2,只能够为1,大于1的话,表示锁定未成功
注2:判断锁是否过期,如果已经过期,当这种情况发生时,不能只是调用DEL来释放锁,然后基于SETNX获取锁定,因为这里有一个竞争关系,当多个客户端检测到一个过期的锁,并均释放锁,然后获取,则都获得了锁定。
17、在判断锁是否过期,如果已经过期,这处代码段中,采用第二种算法,且将file_put_contents全部取消注释,如图20
18、重复第13、14、15等3个步骤,查看/api/runtime目录下所生成的文件,以0、6开头的文件数量为10,以1开头的文件数量为7,以2开头的文件数量为6,以3开头的文件数量为1,以4开头的文件数量为3,以5开头的文件数量为1,正常,如图21
注1:以1开头的文件数量为7,以2开头的文件数量为6,以3开头的文件数量为1,后两者相加正好等于前者,表示在已经过期的7个线程中,只有一个获得了锁定,最终以5开头的文件数量也为1
注2:由于GETSET的特性,可以检查存储在 key 的旧值是否仍然是过期的时间戳,如果是,则获取锁,否则返回假
19、清理掉所有方便于开发期间测试的代码,如file_put_contents等,如图22
// 设置锁定的过期时间,获取相关锁定参数 $time = time(); $lockKey = Yii::$app->params['lock']['keyPrefix'] . 'game_category'; $lockExpire = $time + Yii::$app->params['lock']['timeOut']; // 获取 Redis 连接,以执行相关命令 $redis = Yii::$app->redis; // 获取锁定 $executeCommandResult = $redis->setnx($lockKey, $lockExpire); // 返回0,表示已经被其他客户端锁定 if ($executeCommandResult == 0) { // 防止死锁,获取当前锁的过期时间 $lockCurrentExpire = $redis->get($lockKey); // 判断锁是否过期,如果已经过期 if ($time > $lockCurrentExpire) { // 防止并发锁定,检查存储在 key 的旧值是否仍然是过期的时间戳,如果是,则获取锁定,否则返回假 $executeCommandResult = $redis->getset($lockKey, $lockExpire); if ($lockCurrentExpire != $executeCommandResult) { return ['status' => false, 'code' => 0, 'message' => '']; } } // 返回0,表示已经被其他客户端锁定,且不存在死锁,返回假 if ($executeCommandResult == 0) { return ['status' => false, 'code' => 0, 'message' => '']; } }
20、将获取锁定与释放锁定抽象为一个类文件,/common/models/redis/Lock.php,如图23、24
<?php namespace common/models/redis; use Yii; /** * This is the model class for table "{{%lock}}". * */ class Lock extends /yii/redis/ActiveRecord { /** * Redis模型的锁定实现 * @param string $lockKeyName 锁定键名 * 格式如下: * * 'game_category' //锁定键名,如比赛分类 * * @return integer 成功返回对象数组/失败返回错误信息 * 格式如下: * * [ * 'status' => true //状态 * ] * * 或者 * * [ * 'status' => false, //状态 * 'code' => 0, //返回码 * 'message' => '', //说明 * ] * */ public function lock($lockKeyName) { // 设置锁定的过期时间,获取相关锁定参数 $time = time(); $lockKey = Yii::$app->params['lock']['keyPrefix'] . $lockKeyName; $lockExpire = $time + Yii::$app->params['lock']['timeOut']; // 获取 Redis 连接,以执行相关命令 $redis = Yii::$app->redis; // 获取锁定 $executeCommandResult = $redis->setnx($lockKey, $lockExpire); // 返回0,表示已经被其他客户端锁定 if ($executeCommandResult == 0) { // 防止死锁,获取当前锁的过期时间 $lockCurrentExpire = $redis->get($lockKey); // 判断锁是否过期,如果已经过期 if ($time > $lockCurrentExpire) { // 防止并发锁定,检查存储在 key 的旧值是否仍然是过期的时间戳,如果是,则获取锁定,否则返回假 $executeCommandResult = $redis->getset($lockKey, $lockExpire); if ($lockCurrentExpire != $executeCommandResult) { return ['status' => false, 'code' => 0, 'message' => '']; } } // 返回0,表示已经被其他客户端锁定,且不存在死锁,返回假 if ($executeCommandResult == 0) { return ['status' => false, 'code' => 0, 'message' => '']; } } } /** * Redis模型的释放锁定实现 * @param string $lockKeyName 锁定键名 * 格式如下: * * 'game_category' //锁定键名,如比赛分类 * * @return integer 被删除的keys的数量 * 格式如下: * * 1 //被删除的keys的数量 * * 或者 * * 0 //被删除的keys的数量 * */ public function unlock($lockKeyName) { // 获取相关锁定参数 $lockKey = Yii::$app->params['lock']['keyPrefix'] . $lockKeyName; // 获取 Redis 连接,以执行相关命令 $redis = Yii::$app->redis; // 释放锁定 return $redis->del($lockKey); } }
21、编辑文件:/api/models/redis/GameCategory.php,如图25、26、27
/* Redis模型的锁定实现 */ $lockKeyName = 'game_category'; $lock = new Lock(); $lockResult = $lock->lock($lockKeyName); // 返回 false,表示已经被其他客户端锁定 if ($lockResult['status'] === false) { return ['status' => false, 'code' => 0, 'message' => '']; }
原创文章,作者:ItWorker,如若转载,请注明出处:https://blog.ytso.com/250372.html