美文网首页
Redis 高级功能

Redis 高级功能

作者: alexlee666 | 来源:发表于2019-12-27 15:53 被阅读0次

    一、Redis 管道技术

    1.1 背景

    想象一下,现在需要向 Redis 中添加大量的 KV 对,可以通过直接调用大量的 set 命令来完成,比如:

    127.0.0.1:6379> SET Key0 Value0
    127.0.0.1:6379> SET Key1 Value1
    ......
    127.0.0.1:6379> SET KeyN ValueN
    
    

    然而这种方法存在一个问题:
    Redis-server 在处理每个 command 时,都存在一个 round-trip,所以如果每个 command 都单独传送的话必定会产生大量的 round-trip,这极大地影响了Redis 性能。
    通过管道技术就可以避免上述问题。管道技术通过如下方式来实现:
    多个 Redis 命令进行"打包",然后通过管道一同发送给 Redis-server,然后一同得到响应结果,以提升 Redis 处理效率。

    客户端向服务端发送一个查询请求,并监听Socket返回,通常是以阻塞模式,等待服务端响应。这个过程如同tcp的调用同步化(异步阻塞),管道就是为了优化这种情况

    注意:RTT(Round-Trip Time): 往返时延。在计算机网络中它是一个重要的性能指标,表示从发送端发送数据开始,到发送端收到来自接收端的确认(接收端收到数据后便立即发送确认),总共经历的时延。

    管道命令:redis-cli --pipe

    1.2 举例

    使用 Redis 管道 pipe 的常用步骤:

    1. 将要执行的所有命令写入一个 txt 文件
    2. 将 txt 文件中的命令塞入管道 pipe
    3. 等待处理结果。
    $ cat data.txt 
    SET Key0 Value0
    SET Key1 Value1
    SET KeyN ValueN
    
    
    # 在 shell 中执行命令
    cat data.txt | redis-cli --pipe
    
    
    # 命令执行结束后会打印如下信息:
    All data transferred. Waiting for the last reply...
    Last reply received from server.
    errors: 0, replies: 3
    
    

    Redis 中查看执行结果:

    127.0.0.1:6379> keys *
    1) "Key1"
    2) "Key0"
    3) "KeyN"
    
    

    二、Redis 事务和管道的区别

    Redis 中的事务和 Oracle 中的类似,具有原子性(要么彻底完成,要么不做)和顺序性(按命令请求的先后顺序执行)。
    Redis 中事务和管道的主要区别在于:

    • 管道:client 端行为,将多个命令打包一同(而不是先发一个,再接着发下一个命令)发给 server端,并监听 Socket 返回,通常以阻塞形式等待服务器处理结果;pipe 中的命令一旦到达server根据顺序就会被立即执行;
    • 事务:server 端行为,是指 server 对于client 发过来的多个请求命令进行批处理。以 multi 命令开启事务,接着逐个接收来自 client 端的命令(放入队列queue中),server 会等待直至使用命令exec,才 开始批处理这段时间区间内所接收的所有命令。

    Redis 事务举例:

    127.0.0.1:6379> multi
    OK
    127.0.0.1:6379> set k1 v1
    QUEUED
    127.0.0.1:6379> set k2 v2
    QUEUED
    127.0.0.1:6379> set k3 v3
    QUEUED
    127.0.0.1:6379> exec
    1) OK
    2) OK
    3) OK
    127.0.0.1:6379> keys *
    1) "Key1"
    2) "Key0"
    3) "KeyN"
    4) "k1"
    5) "k3"
    6) "k2"
    
    

    三、Redis 分布式锁

    首先需要明白这里的“锁”,并不是真正意义上的锁。Redis 的 setnx命令具有如下作用:

    • 当 key 存在时,返回 0,不更新其 value;
    • 当 key 不存在时,插入 key-vlaue。

    先拿· setnx 来争抢锁,抢到之后,再用expire给锁加一个过期时间防止锁忘记了释放。 这时候对方会告诉你说你回答得不错,然后接着问如果在 setnx 之后执行 expire 之前进程意外crash或者要重启维护了,那会怎么样? 这时候你要给予惊讶的反馈:唉,是喔,这个锁就永远得不到释放了。紧接着你需要抓一抓自己得脑袋,故作思考片刻,好像接下来的结果是你主动思考出来的,然后回答:我记得set指令有非常复杂的参数,这个应该是可以同时把setnx和expire合成一条指令来用`的!对方这时会显露笑容,心里开始默念:摁,这小子还不错。
    jedis.set(String key, String value, String nx, String expx, int time),这个set()方法一共有五个形参:

    第一个为key,我们使用key来当锁,因为key是唯一的。
    第二个为value,我们传的是requestId,很多童鞋可能不明白,有key作为锁不就够了吗,为什么还要用到value?原因就是我们在上面讲到可靠性时,分布式锁要满足第四个条件解铃还须系铃人,通过给value赋值为requestId,我们就知道这把锁是哪个请求加的了,在解锁的时候就可以有依据。requestId可以使用UUID.randomUUID().toString()方法生成。
    第三个为nx,这个参数我们填的是NX,意思是SET IF NOT EXIST,即当key不存在时,我们进行set操作;若key已经存在,则不做任何操作;
    第四个为expx,这个参数我们传的是PX,意思是我们要给这个key加一个过期的设置,具体时间由第五个参数决定。
    第五个为time,与第四个参数相呼应,代表key的过期时间。


    四、分区

    分区的目的是让多个 Redis 实例能够同时各自处理一份大数据中的分块(通过增加机器、内存),从而提高Redis的处理性能。分为:

    • 范围分区;
    • 哈希分区;

    4.1 范围分区

    映射一定范围的对象到特定的Redis实例。比如,ID从0到10000的用户会保存到实例R0,ID从10001到 20000的用户会保存到R1,以此类推。
    这种方式是可行的,并且在实际中使用,不足就是要有一个区间范围到实例的映射表。这个表要被管理,同时还需要各 种对象的映射表,通常对Redis来说并非是好的方法。

    4.2 哈希分区

    • 用一个 hash 函数将 key 转换为一个数字;
    • 对这个整数取模,将其转化为 0-num 之间的数字(其中 num+1 是 redis 实例数或者分区数);

    即 hash(key) % num。


    五、Redis 集群、分区和哨兵的关系

    Redis Sentinal着眼于高可用,在master宕机时会自动将slave提升为master,继续提供服务。
    Redis Cluster着眼于扩展性,在单个redis内存不足时,使用Cluster进行分片存储。


    https://www.cnblogs.com/silyvin/p/11559433.html
    https://redis.io/topics/mass-insert
    https://www.jianshu.com/p/84b655a55bf5
    https://blog.csdn.net/weixin_42740530/article/details/100940519

    相关文章

      网友评论

          本文标题:Redis 高级功能

          本文链接:https://www.haomeiwen.com/subject/gzzroctx.html