队列操作

简单队列

利用List数据结构可以实现简单的队列,在于使用List提供插入和移除api来完成简单队列操作;

准备数据

获取数据

后入先出

使用redis提供的apiLPOP可以从队列左边获取数据,也就是从队列的头部获取到最后插入的数据,也可以使用BLPOP设置超时时间,从而做到等待数据写入;

先入先出

与上面的一样,只是把获取数据的方向改成另一边,使用RPOP可以从队列右边获取数据,也就是从队列的尾部获取到最先插入的数据,也可以使用BRPOP设置超时时间,从而做到等待数据写入;

等待

使用BLPOPBRPOP可以移除并获取列表头部/尾部获取第一个元素;如果把BLPOP命令的超时时间设置为0,表示一直阻塞,直到数据写入;

实现

到这就可以将List当成一个简单的队列,推送数据的生产者,将数据写入到List,然后消费者等着List中有数据,然后获取;

生产者

using StackExchange.Redis;

var connection =  ConnectionMultiplexer.Connect("localhost");
var redis = connection.GetDatabase(1); foreach (var i in Enumerable.Range(0,10))
{
await redis.ListLeftPushAsync("data",i);
Console.WriteLine($"写入 {i}");
Thread.Sleep(1000);
}

消费者

using StackExchange.Redis;

var connection = ConnectionMultiplexer.Connect("localhost");
var redis = connection.GetDatabase(1); while (true)
{
var res = await redis.ListRightPopAsync("data"); if (!string.IsNullOrEmpty(res))
{
Console.WriteLine($"读取 {res}");
} Thread.Sleep(1000);
}

至此,消息已经消费,但是消息的多播并不能实现,队列中的消息被移除了,另外的消费者就拿不到这条消息,同时ListRightPopAsync方法对应的redisRPOP命令,StackExchange.Redis中不支持Redis中对应的BLPOPBRPOP命令,但是可以使用发布订阅模式实现;

发布订阅

生产者

using StackExchange.Redis;

var connection = ConnectionMultiplexer.Connect("localhost");
var sub = connection.GetSubscriber(); foreach (var i in Enumerable.Range(0,10))
{
await sub.PublishAsync("channel", $"消息 {i}");
Console.WriteLine($"发布消息 {i}");
Thread.Sleep(1000);
} Console.ReadLine();

消费者

using StackExchange.Redis;

var connection = ConnectionMultiplexer.Connect("localhost");
var sub = connection.GetSubscriber(); sub.Subscribe("channel",
(channel, message) =>
{
Console.WriteLine($"接收消息:{message}");
}); Console.ReadLine();

通过消息生产者将消息发布到某个channel,同时消费者订阅该channel,即可获取到消息;当有多个消费者也可以接受到消息了,如果在消费者订阅之前消息就已消费则不能再获取到该消息;

开启两个消费者,都订阅了同一channel,利用Sleep使消费者2晚一秒订阅到channel,导致没有接收到消息0;

如果想要做到消息的持久化可以同时将Redis的List利用起来,或者使用其他介质存一份,在消息发布的同时使用将消息推到List中,然后在消费者中去将消息移除,历史消息可以在订阅channel前获取到当前List中消息,开启一个线程去执行业务操作并移除当前消息;

生产者

using StackExchange.Redis;

var connection = ConnectionMultiplexer.Connect("localhost");
var redis = connection.GetDatabase(1);
var sub = connection.GetSubscriber(); foreach (var i in Enumerable.Range(0,100))
{
await redis.ListLeftPushAsync("historyMsg", $"消息 {i}");
await sub.PublishAsync("channel", $"消息 {i}");
Console.WriteLine($"发布消息 {i}");
Thread.Sleep(1000);
} Console.ReadLine();

消费者

using StackExchange.Redis;

var connection = ConnectionMultiplexer.Connect("localhost");
var redis = connection.GetDatabase(1);
var sub = connection.GetSubscriber(); //这里可以选择不同消息持久化介质
var count = (await redis.ListRangeAsync("historyMsg")).Length; if (count > 0)
{
await Task.Run(() =>
{
for (int i = 0; i < count; i++)
{
string result = redis.ListRightPop("historyMsg");
Console.WriteLine("处理历史消息:" + result);
}
});
} sub.Subscribe("channel",
(channel, message) =>
{
redis.ListRightPop("historyMsg");
Console.WriteLine($"接收消息:{message}");
}); Console.ReadLine();

使用Reids实现简单消息队列的更多相关文章

  1. simple简单消息队列

    一:介绍 1.优缺点 简单,但是耦合性较高. 这种模式是生产者与消费者一一对应,就是一个产生者,有一个消费者来消费. 如果,多个消费者想消费一个队列中的消息就不适合了.这种情况在后面会接着介绍. 2. ...

  2. Redis实现简单消息队列

    http://www.jianshu.com/p/9c04890615ba 任务异步化 打开浏览器,输入地址,按下回车,打开了页面.于是一个HTTP请求(request)就由客户端发送到服务器,服务器 ...

  3. 用 Redis 实现 PHP 的简单消息队列

    参考:PHP高级编程之消息队列 消息队列就是在消息的传输过程中,可以保存消息的容器. 常见用途: 存储转发:异步处理耗时的任务 分布式事务:多个消费者消费同一个消息队列 应对高并发:通过消息队列保存任 ...

  4. redis(五)---- 简单消息队列

    消息队列一个消息的链表,是一个异步处理的数据处理引擎.不仅能够提高系统的负荷,还能够改善因网络阻塞导致的数据缺失.一般用于邮件发送.手机短信发送,数据表单提交.图片生成.视频转换.日志储存等. red ...

  5. RabbitMq(2) 简单消息队列

    <dependency> <groupId>com.rabbitmq</groupId> <artifactId>amqp-client </ar ...

  6. redis简单消息队列

    <?php $redis = new Redis(); $redis->connect('127.0.0.1',6379); $redis->flushall(); $redis-& ...

  7. Redis使用总结(3):实现简单的消息队列

    参考Redis实现简单消息队列 Redis提供了两种方式来作消息队列.一个是使用生产者消费模式模式,另外一个方法就是发布订阅者模式.前者会让一个或者多个客户端监听消息队列,一旦消息到达,消费者马上消费 ...

  8. 玩转redis-简单消息队列

    使用go语言基于redis写了一个简单的消息队列 源码地址 使用demo redis的 list 非常的灵活,可以从左边或者右边添加元素,当然也以从任意一头读取数据 添加数据和获取数据的操作也是非常简 ...

  9. PDO和消息队列的一点个人理解

    什么是消息队列,百度百科说,···消息队列····是在消息的传输过程中保存消息的容器. 看着网上林林总总的文章,都说是为了应对高并发,处理数据量超级大的一种数据容器,也可以说是利用各种方式,先把数据存 ...

  10. RocketMQ 消息队列单机部署及使用

    转载请注明来源:http://blog.csdn.net/loongshawn/article/details/51086876 相关文章: <RocketMQ 消息队列单机部署及使用> ...

随机推荐

  1. 从零开始学java(第二天)

    ------------恢复内容开始------------ 今天是学习了一些基础的知识 1.注释 //行注释 /*多行注释*/ /**文档注释*/ 2.标识符和关键字 标识符就是名字,类名方法名变量 ...

  2. cmu15545笔记-并发控制总结(Concurrency Control Summary)

    目录 总览 ACID 串行化与冲突操作 隔离级别 概念层级 二阶段锁 原理 级联回滚 强二阶段锁 死锁检测和避免 锁层级 实践应用 实现的隔离级别 OOC 原理 三个阶段 实现的隔离级别 处理幻读 M ...

  3. 德承工控机DX-1200 成功适配2024年6月6日发布的国产开源系统OpenEuler 24.03 LTS

    基础软件双子星:欧拉系统(OpenEuler)& 鸿蒙系统(OpenHarmony),鸿蒙系统常应用在华为的手机和平板电脑上,大众也较为熟悉,是面向消费电子产品领域的系统:而欧拉系统则是面向服 ...

  4. nrm安装后无法使用

    前情 在使用node.js的过程中,经常会时不是遇到有些包下载安装慢或者失败,有时可以尝试切换源来解决这类问题 坑 通过npm install nrm -g安装完nrm后运行nrm一直报错 Why? ...

  5. Linux行号显示

    xshell显示行号: 输入命令: vim ~/.vimrc 输入: set nu 之后在打开文件 就可以 看到行号显示.

  6. 关于 Envoy on Windows

    Window Image in hub.cocker.com envoy 的镜像位于 https://hub.docker.com/u/envoyproxy 之下,其中 Windows 包括如下 4 ...

  7. metasploit模块

    模块类型 辅助模块(Auxiliary) 渗透攻击模块(Exploits) 后渗透攻击模块(Post) 攻击载荷模块(Payloads) 空指令模块(Nops) 编码器模块(Encoders) 后渗透 ...

  8. 【Vue】Vue脚手架安装搭建

    ## Vue脚手架安装搭建 操作系统:Windows 11 专业工作站 记录时间:2022年2月18日 在安装之前,需要安装node.js(Node.js (nodejs.org)),否则无法使用np ...

  9. Server check fail, please check server xxx ,port 9848 is available

    [1]如果使用docker安装的nacos服务,2.x版本后增加了 grpc 通信并且增加nacos的集群端口上下偏移1000,创建容器时除了8848还需要把7848.9848都暴露出来.如:-p 7 ...

  10. IDEA和GIT关于文件中LF和CRLF问题

    问题描述:项目软件安装shell脚本上git仓库管理,但拉取后,上linux运行报错. 问题思考:根据描述信息可以查看到\r字样,初步判别为换行符导致 1.将脚本文件移动至notepad++中,通过视 ...