《Redis 分布式锁》
一:什么是分布式锁。
- 通俗来说的话,就是在分布式架构的redis中,使用锁。
二:分布式锁的使用选择。
- 当 Redis 的使用场景不多,而且也只是单个在用的时候,可以构建自己使用的 锁。
- 如果在公司里落地生产环境用分布式锁的时候,一般是会用开源类库。
- Redis分布式锁,一般就是用 Redisson 框架就好了,非常的简便易用。
三:Redisson 实现Redis分布式锁的原理。
- 整体流程图
- 
- 详解
- 当某个请求的客户端需要加锁时候。
- 如果该客户端面对的是一个redis cluster集群,他首先会根据hash算法选择一台机器。(一致hash算法)
- 随后会向Redis发送加锁请求(原理为Lua脚本)
- Lua 脚本会保证业务执行的原子性,所以采用 Lua 的方式为加锁命令。
- 
- 这个Lua 脚本的大致意思为(这个Key的锁是否存在,如果存在,则一直循环等待,如不存在,则上锁,并设置过期时间。)
- watch dog 提供的锁自动延期的能力
- 加锁的锁key默认生存时间才30秒,如果超过了30秒,客户端1还想一直持有这把锁,怎么办呢?
- RedisSon 只要加锁成功,就会启动一个watch dog看门狗, 他是一个后台线程,会每隔10秒检查一下 ,如果还持有锁key,那么就会不断的延长锁key的生存时间。
四:分布式锁锁带来的问题
- 在分布式架构中的问题,就是如果你对某个redis master实例,写入了key的锁,此时会异步复制给对应的master slave实例。
- 但是这个过程中一旦发生redis master宕机,主备切换,redis slave变为了redis master。
- 接着就会导致,客户端2来尝试加锁的时候,在新的redis master上完成了加锁,而客户端1也以为自己成功加了锁。
- 此时就会导致多个客户端对一个分布式锁完成了加锁。
- 这时系统在业务语义上一定会出现问题, 导致各种脏数据的产生 。
- 这个就是redis cluster/redis master-slave架构的 主从异步复制 导致的redis分布式锁的问题:在redis master实例宕机的时候,可能导致多个客户端同时完成加锁。
五:简单实现的redis锁
- 原理
- 通过 redis 的 setnx 功能进行加锁
- 实现(PHP)
<?php
/**
* 创建 Redis 单例
* @return mixed
*/
class redisInstance
{
static $redis; private function __construct()
{
} public static function getInstance()
{
if (self::$redis) {
return self::$redis;
} self::$redis = new \Redis(); return self::$redis;
} } class redisLock
{
public $reids; public function __construct()
{
$this->reids = redisInstance::getInstance();
} /**
* 键加锁,默认加锁时间为 1分钟 = 60 * 1000(毫秒)
* @param $key
* @param int $ttl
* @return bool
*/
public function lock($key, $ttl = 60000)
{
$lockKey = $key . '_lock'; // 通过setNx命令拿到锁
$lock = $this->reids->set($lockKey, 1, ['NX', 'PX' => $ttl]); // 拿到锁则直接返回
if ($lock) {
return true;
} // 没有拿到锁,则一直循环等待锁资源释放
while (!$lock) {
$lock = $this->reids->set($key, 1, ['NX', 'PX' => $ttl]);
} return $lock;
} /**
* 释放锁
* @param $key
* @return bool
*/
public function unLock($key)
{
$lockKey = $key . '_lock'; $lock = false; // 释放锁
while (!$lock) {
$lock = $this->reids->del($lockKey);
} return true;
}
} $redis = redisInstance::getInstance();
$redisLock = new redisLock(); /**
* 例如,买商品,进行库存递减
* 1:对库存加锁
* 2:递减
* 3:释放锁
* 影响
* 加锁导致的并发度降低
*/
$key = 'stock';
$redisLock->lock($key);
$stock = $redis->get($key);
if ($redis->get($key) <= 0) {
$redisLock->unLock($key);
return false;
}
$redis->decr($key);
$redisLock->unLock($key);
《Redis 分布式锁》的更多相关文章
- 简单物联网:外网访问内网路由器下树莓派Flask服务器
最近做一个小东西,大概过程就是想在教室,宿舍控制实验室的一些设备. 已经在树莓上搭了一个轻量的flask服务器,在实验室的路由器下,任何设备都是可以访问的:但是有一些限制条件,比如我想在宿舍控制我种花 ...
- 利用ssh反向代理以及autossh实现从外网连接内网服务器
前言 最近遇到这样一个问题,我在实验室架设了一台服务器,给师弟或者小伙伴练习Linux用,然后平时在实验室这边直接连接是没有问题的,都是内网嘛.但是回到宿舍问题出来了,使用校园网的童鞋还是能连接上,使 ...
- 外网访问内网Docker容器
外网访问内网Docker容器 本地安装了Docker容器,只能在局域网内访问,怎样从外网也能访问本地Docker容器? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Docker容器 ...
- 外网访问内网SpringBoot
外网访问内网SpringBoot 本地安装了SpringBoot,只能在局域网内访问,怎样从外网也能访问本地SpringBoot? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装Java 1 ...
- 外网访问内网Elasticsearch WEB
外网访问内网Elasticsearch WEB 本地安装了Elasticsearch,只能在局域网内访问其WEB,怎样从外网也能访问本地Elasticsearch? 本文将介绍具体的实现步骤. 1. ...
- 怎样从外网访问内网Rails
外网访问内网Rails 本地安装了Rails,只能在局域网内访问,怎样从外网也能访问本地Rails? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Rails 默认安装的Rails端口 ...
- 怎样从外网访问内网Memcached数据库
外网访问内网Memcached数据库 本地安装了Memcached数据库,只能在局域网内访问,怎样从外网也能访问本地Memcached数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装 ...
- 怎样从外网访问内网CouchDB数据库
外网访问内网CouchDB数据库 本地安装了CouchDB数据库,只能在局域网内访问,怎样从外网也能访问本地CouchDB数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Cou ...
- 怎样从外网访问内网DB2数据库
外网访问内网DB2数据库 本地安装了DB2数据库,只能在局域网内访问,怎样从外网也能访问本地DB2数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动DB2数据库 默认安装的DB2 ...
- 怎样从外网访问内网OpenLDAP数据库
外网访问内网OpenLDAP数据库 本地安装了OpenLDAP数据库,只能在局域网内访问,怎样从外网也能访问本地OpenLDAP数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动 ...
随机推荐
- java 线程安全(初级)
创建和启动Java线程 Java线程是个对象,和其他任何的Java对象一样.线程是类的实例java.lang.Thread,或该类的子类的实例.除了对象之外,java线程还可以执行代码. 创建和启动线 ...
- jupyter Notebook 设置密码
由于服务器关闭了图形界面 所以在服务器上安装Jupyter Notebook 随后本机web访问,利用本机的显卡可以执行plt相关图形命令 本次介绍如何设置Jupyter Notebook的密码设定 ...
- 洛谷 P4017 最大食物链计数 题解
P4017 最大食物链计数 题目背景 你知道食物链吗?Delia生物考试的时候,数食物链条数的题目全都错了,因为她总是重复数了几条或漏掉了几条.于是她来就来求助你,然而你也不会啊!写一个程序来帮帮她吧 ...
- Calibre中使用DeDRM插件进行Kindle电子书解锁
小书匠 废话不多说,下面是Calibre和DeDRM插件的下载地址: https://calibre-ebook.com/download https://github.com/apprenticeh ...
- Linux环境下Nexus3.6安装
1. 安装JDK 2. 下载nexus开源版本即可,Nexus OSS下载 流程 3. 解压文件,会的得到两个文件夹[nexus-3.6.0]和[sonatype-work] tar -zxv ...
- debian10使用国内源安装docker以及一些使用方法
首先, 我的环境是debian, 容器是centos debian 安装添加新存储库所需的依赖项 1 sudo apt install ca-certificates curl software-pr ...
- 【Java 8】巧用Optional之优雅规避NPE问题
避之不及的 NullPointerException NPE : NullPointerException 空指针异常是最常见的Java异常之一,抛出NPE错误不是用户操作的错误,而是开发人员的错误, ...
- Healthcare in Azure
- mysql事务回滚机制概述
应用场景: 银行取钱,从ATM机取钱,分为以下几个步骤 1 登陆ATM机,输入密码: 2 连接数据库,验证密码: 3 验证成功,获得用户信息,比如存款余额等: 4 用 ...
- max函数比较字符串类型
关于sql中 max函数比较字符串类型 max只比较首个字符的大小 只要首字母大,则不比较其他位置的字母,若首字母相同,则比较顺序位字母. 今天死在这了 数据库中 step字段类型char分别为 5. ...