一、Redis数据类型

  1.string

    string是Redis的最基本数据类型,一个key对应一个value,每个value最大可存储512M。string一半用来存图片或者序列化的数据。

  2.hash

    相当于一个string类型的映射表。特别适合用来存储对象。例如可以存储用户信息,用户ID作为hash类型里的每一个key。

    案例:我们这边需要对接微信粉丝的数据到我们自己的平台上,但微信提供的接口只支持单天查询,那么如果我们想要查看最近一个月微信粉丝的状况,就需要循环30次调用微信的接口。一个月勉强还可以接受,那么如果想要查半年,甚至一年呢?那么我们的接口里就需要循环365次调微信的接口,这就会使我们的接口变得非常慢,甚至超时。还有这些数据,比如单天新增粉丝数,是不会变得,而且每天都有一个数据,这样就特别适合存在redis的hash类型里,以日期(2018-10-10)作为hash的key。

  3.list

    list类型是简单的字符串列表,每个列表可以存储232 - 1 个值。可以从头部或者尾部顺序插入数据。list类型可以用来做电商里的秒杀营销系统或关注列表。

  4.set

    set是string类型的无序集合。该集合是通过哈希实现的,添加、删除的复杂度都是O(1),所以查找非常快。

    案例:我们这边是以手机号为唯一标示符,防止重复用户注册,会判断该手机号有没有注册过,那么如果用set类型存储注册过的用户手机号,就会很快判断出该用户是否注册过,而不用去查数据库了。

  5.zset

    和set一样,但zset多了一个score来让set变得有序,且不允许有重复的成员。

    案例:我们这边有一个账户记录需要按记录时间排序,那么就可以将时间戳当作score存储zset中。

二、redis分布式锁

  网上很多redis分布式锁的实现方式不能说错误的,但至少不够严谨,在某些极端情况下是会出问题的。一旦出现问题,还是挺麻烦的事情,所以我们要知道redis分布式锁的正确姿势。

  其实很简单,利用redis的原子性。关于原子性,官方的一段描述为:

  大概意思就是redis执行lua脚本的时候,会被当成一条命令执行,在此期间,不会执行其他命令,所以lua脚本尽量是快脚本而不是慢脚本。

  所以,正确的姿势是:

    public function getDistributeLock($redis, string $key, int $userId, int $expire)
{
$luaScript = <<<LUA
if (redis.call('exists', KEYS[1]) == 0) and redis.call('setex', KEYS[1], ARGV[1], ARGV[2])
then
return 1
else
return 0
end
LUA; return $redis->eval($luaScript, 1, $key, $expire, $userId) > 0 ? true : false;
}

  这里我们把一段lua脚本放到redis的eval方法里执行,这样就可以保证这一段命令的原子性。

  那么如果不用lua脚本,姿势应该是这样的:

    public function wrongGetDistributeLock($redis, string $key, int $userId, int $expire)
{ // 若锁不存在,则加锁
if(!$redis->exists($key) && ($redis->setex($key, $expire, $userId) == 'OK')) {
return true;
} return false;
}

  前面提到过,这种姿势在某些情况下会出问题:如果同时好几个客户端同时请求,同时通过了上面if条件的第一层,那么这时候就会出现多个同时拿到锁,并且前面人的锁会被覆盖。

  然后,正确的解锁姿势是:

    public function releaseDistributeLock($redis, string $key, int $userId)
{
$luaScript = <<<LUA
if redis.call('get', KEYS[1]) == ARGV[1]
then
return redis.call('del', KEYS[1])
else
return 0
end
LUA;
$redis->eval($luaScript, 1, $key, $userId);
}

  同样需要使用lua脚本来达到原子性。那么如果不使用lua脚本的姿势是:

    public function wrongReleaseLock($redis, string $key, int $userId)
{
if($userId == $redis->get($key)) {
$redis->del($key);
}
}

  会有这样一种情况:A请求通过if语句后,这时候这个redis的key刚好过期了,然后B客户端加锁成功,这时候A请求就会把客户端B刚加的锁给解除了。

  虽然我上面提到的两种情况都是很极端、很少出现的。但如果可以用很简单的方法避免掉,so why not?

  上面提到的redis分布式锁,满足了三个特性:

  • 互斥性。同时只能又一个客户端拥有锁
  • 不会发生死锁。 
  • 加锁和解锁的必须是同一个客户端。

  童鞋们,有什么疑问,可以在地下留言哦。

Redis全方位详解--数据类型使用场景和redis分布式锁的正确姿势的更多相关文章

  1. Redis全方位详解--磁盘持久化和容灾备份

    序言 在上一篇博客中,博客介绍了redis的数据类型使用场景和redis分布式锁的正确姿势.我们知道一旦Redis重启,存在redis里面的数据就会全部丢失.所以这篇博客中向大家介绍Redis的磁盘持 ...

  2. 【分布式缓存系列】Redis实现分布式锁的正确姿势

    一.前言 在我们日常工作中,除了Spring和Mybatis外,用到最多无外乎分布式缓存框架——Redis.但是很多工作很多年的朋友对Redis还处于一个最基础的使用和认识.所以我就像把自己对分布式缓 ...

  3. 【分布式缓存系列】集群环境下Redis分布式锁的正确姿势

    一.前言 在上一篇文章中,已经介绍了基于Redis实现分布式锁的正确姿势,但是上篇文章存在一定的缺陷——它加锁只作用在一个Redis节点上,如果通过sentinel保证高可用,如果master节点由于 ...

  4. Redis实现分布式锁的正确姿势

    分布式锁一般有三种实现方式:1. 数据库乐观锁:2. 基于Redis的分布式锁:3. 基于ZooKeeper的分布式锁.本篇博客将介绍第二种方式,基于Redis实现分布式锁.虽然网上已经有各种介绍Re ...

  5. 掌握Redis分布式锁的正确姿势

    本文中案例都会在上传到git上,请放心浏览 git地址:https://github.com/muxiaonong/Spring-Cloud/tree/master/order-lock 本文会使用到 ...

  6. Redis分布式锁的正确姿势

    1. 核心代码: import redis.clients.jedis.Jedis; import java.util.Collections; /** * @Author: qijigui * @C ...

  7. Redis学习详解(一):Redis持久化机制之RDB

    Redis的持久化机制有两种:RDB持久化和AOF持久化.因为Redis是一个内存数据库,如果没有合适的持久化机制,那么一旦服务器进程退出,服务器中的数据库状态也会消失.本章介绍RDB持久化机制. R ...

  8. python操作redis用法详解

    python操作redis用法详解 转载地址 1.redis连接 redis提供两个类Redis和StrictRedis用于实现Redis的命令,StrictRedis用于实现大部分官方的命令,并使用 ...

  9. Redis原理详解

    Redis原理详解 数据类型 Redis最为常用的数据类型主要有以下五种: String Hash List Set Sorted set 在具体描述这几种数据类型之前,我们先通过一张图了解下Redi ...

随机推荐

  1. redis 命令select、dbsize、清空数据库、info、client

    select 切换库 dbsize 当前库中数据条数 清空数据库 flushdb | flushall 清空数据,一个是清空当前库,一个清空当前实例 查看服务器及redis相关信息 infoinfo ...

  2. python学习手册中的一些易忘的点(第4部分)

    1.python函数高聚合低耦合1)对于输入使用参数并且对于输出使用return2)只有在真正必要的情况下使用全局变量3)不要改变可变类型的参数,除非调用者希望这样做4)每个函数都应该有一个单一的.统 ...

  3. C++虚函数原理

    类中的成员函数分为静态成员函数和非静态成员函数,而非静态成员函数又分为普通函数和虚函数. Q: 为什么使用虚函数 A: 使用虚函数,我们可以获得良好的可扩展性.在一个设计比较好的面向对象程序中,大多数 ...

  4. npm run dev时报错“events.js:160 throw er; // Unhandled 'error' event”

    经查,此问题由端口占用导致,node服务器默认端口8080已被其他程序占用,关闭占用端口的程序或者修改node服务器的默认端口即可解决此问题

  5. [控件] AngleGradientView

    AngleGradientView 效果 说明 1. 用源码产生带环形渐变色的view 2. 可以配合maskView一起使用 (上图中的右下角图片的效果) 源码 https://github.com ...

  6. 基于NSString处理文件的高级类

    基于NSString处理文件的高级类 我已经把处理文件的类简化到了变态的程度,如果你还有更简洁的方法,请告知我,谢谢! 使用详情: 源码: // // NSString+File.h // Maste ...

  7. java中常用Redis操作

     stringRedisTemplate.opsForValue().set("test", "100",60*10,TimeUnit.SECONDS);//向 ...

  8. Centos7 apache2.4.29(httpd) 安装

    重点参考文章:https://blog.csdn.net/MrDing991124/article/details/78829184  写的很详细了,自己按着改博文走了不遍,不错! 一.配置安装环境 ...

  9. mysql的表和约束操作

    在创建表是默认为加上数据引擎和字符集,如创建一个student表,代码如下: create table students(id int unsigned zerofill auto_increment ...

  10. 线段树&&线段树的创建线段树的查询&&单节点更新&&区间更新

    目录 线段树 什么是线段树? 线段树的创建 线段树的查询 单节点更新 区间更新 未完待续 线段树 实现问题:常用于求数组区间最小值 时间复杂度:(1).建树复杂度:nlogn.(2).线段树算法复杂度 ...