之前我们介绍了一些列关于Redis的数据结构、持久化、过期&淘汰策略、集群化等知识点,感兴趣的小伙伴可以在文章的末尾查看往期内容。今天将为大家带来Redis的应用。由于本篇文章较长,所以将拆分为两章来讲解。

除了最基本的KV缓存外,Redis还能用到以下方面。

分布式锁

在分布式应用中,经常会遇到并发问题需要处理,这个时候往往就需要用到分布式锁,而Redis很好的提供了分布式锁的处理。常用的Redis分布式锁有以下两种方式。

setnx

  • setnx(set if not exists):通过setnx设置key,达到获取锁的目的。处理完逻辑通过del删除key释放锁。

  • 死锁处理:为了避免释放锁失败,导致程序进入死锁,通常采用 给key加上过期时间的方式来处理。

Redis2.8 开始加入了set扩展参数,支持setnx 和 expire可以同时执行,是setnx 和 expire在同一个事务中,避免了expire设置过期时间不成功的问题。

  • 任务超时:设置超时时间虽然能避免死锁,但是会存在因超时而使锁提前释放的可能。所以该方案不适合需要较长时间的任务。为了避免任务超时,可以采取 给key设置随机value的方式来解决该问题。释放锁时,先匹配value是否一致,再删除key。

Redlock

除了setnx外,Redis还提供了Redlock提供Redis分布式锁。

要求:Redlock需要多个Redis实例支持,这些Redis实例要求相互独立,没有主从关系。

实现:加锁时,客户端会向超过半数的节点发送setnx指令,只要过半数节点set成功,则加锁成功。释放锁时,需要向所有节点发送del指令。

使用场景:Redlock适用于对高可用要求高的场景,即使一台Redis挂了,也不会影响分布式锁的实现。同时Redlock需要更多的Redis实例,相较于setnx性能有所下降,同时运维代价更高。

消息队列

Redis中实现消息队列常用的list、zset 及Stream几种数据结构可以实现。

list

Redis可以通过list实现异步消息队列。

  • 实现:用rpush和lpush实现入队操作,lpop和rpop实现出队操作,rpush与lpop,lpush与rpop配合使用;

  • 阻塞读:lpop/rpop存在空队列循环取值的问题,造成不必要的资源浪费。通过使用blpop/brpop可以实现队列的阻塞读。当队列中没有数据时,读操作会进入休眠等待状态,当有数据时,则立即唤醒读操作。

zset

Redis通过zset可以实现延时队列。

  • 实现:将消息内容序列化成字符串,作为zset的value,消息处理的到期时间作为score。然后轮询zet获取到期的任务进行处理。

  • zrem:Redis的zrem(key, value)指令,可以从zset中移除指定value,同时能够避免多线程、多进程之间的资源争抢。

Stream

Redis自5.0起开始支持Stream数据结构,Stream是一个能够支持多播可持久化化的消息队列。

  • 实现

  • Stream为一个消息链表,每个节点包含有一个唯一的消息ID和消息内容两部分内容,消息是持久化的;
  • 每个Stream都可以设置多个消费组组,每个消费组都有一个游标(last_delivered_id),表示当前消费组已经消费到了哪条消息,各个消费组间互不影响;
  • 每个消费组都可以有多个消费者,消费者之间是相互竞争,即同一条消息在同一个消费组内不会被重复消费;
  • PEL(Pending Entries List) —— 每个消费组内都有一个pending_ids,记录当前哪些消息已被客户端读取,当收到客户端返回的ack时,消息ID从pending_ids中排除。以此结构来确保客户端至少消费了一次消息,保证消息不会被丢失。
  • 指令:Steram有以下常用的指令:
  1. xadd: 向Stream中添加消息;
// * 表示服务器自动生成ID,后面分别是 key value
// maxlen 表示设置该stream的最大长度,当消息过长,会丢弃老的消息
127.0.0.1:6379> xadd maxlen 100 javamd * test null
1582983245000-0 // 返回生成的消息ID,1 表示当前时间生成的第一条消息
  1. xdel: 从Stream中逻辑移除消息(设置标记位,并不影响Stream的真实长度);
127.0.0.1:6379> xdel javamd 1582983245000-1
(integer) 1
  1. xrange:获取Stream中未标记的消息列表;
127.0.0.1:6379> xrange javamd 1582983245000-0 + // 1582983245000-0为最小消息ID的列表
1) 1) 1582983245000-2
2) 1) “test”
2)null
1) 1) 1582983245000-0
2) 1) “test”
2)null
127.0.0.1:6379> xrange javamd - 1582983245000-8 // 1582983245000-8为最大消息ID的列表
1) 1) 1582983245000-0
2) 1) “test”
2)null
2) 1) 1582983245000-2
2) 1) “test”
2)null
  1. xlen: 获取Stream中所有消息的长度;
127.0.0.1:6379> xlen javamd
(integer) 8 // 不包含已标记的消息
  1. del:物理删除整个Steram;
127.0.0.1:6379> del javamd
(integer) 1 // 删除整个Stream
  1. xread:在不定义消费组的情况下进行Stream的独立消费;
127.0.0.1:6379> xread count 2 streams javamd 0-0 // 从Stream头部读取两条消息
1) 1) “javamd”
2) 1) 1) 1582983245000-0
2) 1) “test”
2)null
2) 1) 1582983245000-2
2) 1) “test”
2)null

使用xread进行顺序消费时,需要记录当前消费消息的ID。下次调用时,需要将当前消费的消息ID作为参数回传。

xread 可以添加block参数,当参数值大于0时,表示阻塞多少ms,如果相应时间内没有消息到来,返回nil。当参数等于0时,表示永远阻塞,直到新的消息添加到Stream中。

  1. xgroup create: 创建消费组;
127.0.0.1:6379> xgroup create javamd group_javamd1 0-0 // 表示从头部开始消费
OK
127.0.0.1:6379> xgroup create javamd group_javamd2 $ // $ 表示从尾部开始消费,只接受新消息
OK
  1. xreadgroup:通过消费组进行消费。
// > 表示从当前消费组的 last_delivered_id 后面开始读
127.0.0.1:6379> xreadgroup GROUP javamd1 c1 count 1 streams javamd >
1) 1) “javamd”
2) 1) 1) 1582983245000-0
2) 1) “test”
2)null

未完待续……

Redis系列推荐

Redis05——Redis Cluster 如何实现分布式集群

Redis04——五分钟明白Redis的哨兵模式

Redis03——Redis是如何删除你的数据的

Redis02——Redis内存数据如何保存到磁盘

Redis01——Redis究竟支持哪些数据结构

Redis06——Redis到底能用在什么地方(上)的更多相关文章

  1. Redis07——Redis到底能用在什么地方(下)

    在前一篇文章中,我们已经介绍过Redis的一些实际应用.如KV缓存.分布式锁.消息队列,由于篇幅原因,并未介绍完全.接下来将继续为各位带来Redis的更多应用. bitmat(位图) 实现 位图的基本 ...

  2. Redis到底该如何利用?

    Redis是个好东西,经过上两个星期的研究和实践,目前正在项目里大规模的替换掉原来的本地内存cache.但是替换过程中却发现,Redis这东西高端,大气上档次,似乎不是我想象里的使用方法. 在没有深入 ...

  3. Redis到底该如何利用?【转自:http://www.cnblogs.com/capqueen/p/HowToUseRedis.html】

    Redis是个好东西,经过上两个星期的研究和实践,目前正在项目里大规模的替换掉原来的本地内存cache.但是替换过程中却发现,Redis这东西高端,大气上档次,似乎不是我想象里的使用方法. 在没有深入 ...

  4. Redis 数据结构与内存管理策略(上)

    Redis 数据结构与内存管理策略(上) 标签: Redis Redis数据结构 Redis内存管理策略 Redis数据类型 Redis类型映射 Redis 数据类型特点与使用场景 String.Li ...

  5. Redis分布式锁 (图解-秒懂-史上最全)

    文章很长,而且持续更新,建议收藏起来,慢慢读! 高并发 发烧友社群:疯狂创客圈(总入口) 奉上以下珍贵的学习资源: 疯狂创客圈 经典图书 : 极致经典 + 社群大片好评 < Java 高并发 三 ...

  6. Redis到底该如何利用(二)?

    上一篇文章里我简述了使用Keys作为Redis搜索的方式,确实感受到了社区的力量,写文章好处多.首先谢谢各位前辈的指导,我知道了拿Redis作为搜索是个错误的方向.本来这篇文章我觉得确实没必要发了,但 ...

  7. 搞懂Redis到底快在哪里

    前言 Redis是一种基于键值对(Key-Value)的NoSQL数据库,Redis的Value可以由String,hash,list,set,zset,Bitmaps,HyperLogLog等多种数 ...

  8. 四个大点,搞懂 Redis 到底快在哪里

    来源:https://mp.weixin.qq.com/s/4kPlBE3C6lTuSvt5mY5hUQ 前言 一. 开发语言 二. 纯内存访问 三. 单线程 四. 非阻塞多路I/O复用机制 前言 R ...

  9. Redis 到底是单线程还是多线程?我要吊打面试官!

    最近在Java技术栈公众号发布的一篇文章,其中有一道题: Redis是多线程还是单线程?(回答单线程的请回吧,为什么请回,请往下看) 好些粉丝在后台问我:为什么请回,Redis不是单线程吗? 大家注意 ...

随机推荐

  1. 前端-js-长期维护

    ###############    JS简介和JS引入     ################ <!DOCTYPE html> <html lang="en" ...

  2. MAC上安装maven以及配置Intellij IDEA

    大前提:java环境已经配置好 maven是对于java工程的管理 一.maven安装到mac 1.首先,maven下载地址http://maven.apache.org/download.cgi 点 ...

  3. 安装oracle 11g 客户端,检查过程中报物理内存不足的解决

    今早接到同事电话,说安装oracle 11g客户端的时候,在检查先决条件的时候,报错,说内存不足,但是本机的内存是2G,肯定够用:如图: 找了一圈,原来Oracle执行先决条件检查是依赖c$共享,很多 ...

  4. HOG算法基础

    实现思路步骤: 1.对原图像gamma校正,img=sqrt(img); 2.求图像竖直边缘,水平边缘,边缘强度,边缘斜率. 3.将图像每16*16(取其他也可以)个像素分到一个cell中.对于256 ...

  5. RPC 框架性能大比拼

    Dubbo 是阿里巴巴公司开源的一个Java高性能优秀的服务框架,使得应用可通过高性能的 RPC 实现服务的输出和输入功能,可以和 Spring框架无缝集成. Motan 是新浪微博开源的一个Java ...

  6. MyBatisUtil

    package com.it.util; import java.io.IOException; import java.io.Reader; import org.apache.ibatis.io. ...

  7. hql错误:No data type for node: org.hibernate.hql.ast.tree.IdentNode

    今天写了一个查询,用的是hql,数据库是mysql.多表联查,结果报错了报: \-[IDENT] IdentNode: 'routerNumber' {originalText=routerNumbe ...

  8. JavaScript常见排序算法

    1.冒泡排序 function bubble_sort(arr) { if (arr.length <= 1) { return arr; } var len = arr.length; for ...

  9. 网购分期不还 N种恶果等着你

    N种恶果等着你" title="网购分期不还 N种恶果等着你"> 网购市场狂飙突进的发展,让每个人都享受到随时随地购物的乐趣,也在很大程度上推动商品之间的流通.目前 ...

  10. Introduction Of Gradient Descent

    不是一个机器学习算法 是一种基于搜索的优化方法 作用:最小化一个损失函数 梯度上升法:最大化一个效用函数 import matplotlib.pyplot as plt import numpy as ...