Redis的Pub/Sub客户端实现
前言
在学习T-io框架,从写一个Redis客户端开始一文中,已经简单介绍了Redis
客户端的实现思路,并且基础架构已经搭建完成,只不过支持的命令不全,不过后期在加命令就会很简单了。本篇就要实现Publish/Subscribe
功能。
Pub/Sub
发布订阅模式在很多场景中用的都很频繁,这里不再赘述。下面看一下Redis
中的命令。参考资料:https://redis.io/topics/pubsub
//发布
PUBLISH
//订阅
SUBSCRIBE
//模式匹配订阅
PSUBSCRIBE
//取消订阅
UNSUBSCRIBE
//取消订阅(模式匹配)
PUNSUBSCRIBE
//其他
PUBLISH/SUBSCRIBE
命令使用方式很简单:
PUBLISH CHANNEL MESSAGE
例如:publish user helloworld
Client
类中增加代码:
@Override
public void publish(final String channel, final String message) {
sendCommand(Command.PUBLISH, channel, message);
}
@Override
public void subscribe(final String... channels) {
sendCommand(SUBSCRIBE, channels);
}
调试代码如下:
//发布
Tedis tedisPublish = new Tedis("192.168.1.225", 6379);
tedisPublish.publish("channel1","hello world");
//订阅
Tedis tedis = new Tedis("192.168.1.225", 6379);
tedis.subscribe(new MyPubSub(),"channel1");
先订阅,后发布,订阅响应结果:
*3
$9
subscribe
$8
channel1
:1
通过响应结果可以看出,我们当前的命令是 subscribe
,然后订阅的是channel1
,当前共订阅了:1
个。
发布响应结果:
:1
总共发给了:1
个订阅客户端。这个结果就是订阅客户端的个数。
PSUBSCRIBE
命令格式如下:
PSUBSCIRBE news.*
修改一下调试代码:
订阅
tedis.pSubscribe(new MyPubSub(),"news.*");
响应结果:
*3
$10
psubscribe
$6
news.*
:1
发布
tedisPublish.publish("news.sports","welcome to NBA");
tedisPublish.publish("news.country","this is china news");
订阅客户端收到消息:
*4
$8
pmessage
$6
news.*
$11
news.sports
$14
welcome to NBA
*4
$8
pmessage
$6
news.*
$12
news.country
$18
this is china news
从响应结果可以看出,客户端订阅了 news.*
,然后收到了news.sports,news.country
的消息。
响应消息解析
上述代码中有一个MyPubSub
对象,它继承自抽象类TedisPubSub
.这个类做了发布订阅核心的业务处理。通过对服务端返回的消息格式,我们可以发现,它的消息格式是统一的。
EVENT_NAME --事件
CHANNEL_NAME --频道
OTHER --其他信息,根据每个事件可能不同
所以我们在做发布订阅的响应消息解析时,可以返回 List。这里以SUBSCRIBE/PSUBSCRIBE
举例
private boolean handleSubscribe(byte[] resp,List<Object> reply){
//是否普通订阅
boolean isSubscribe = Arrays.equals(SUBSCRIBE.raw, resp);
//是否模式匹配订阅
boolean isPSubscribe = Arrays.equals(Keyword.PSUBSCRIBE.raw, resp);
if (isSubscribe || isPSubscribe) {
resetSubscribedChannels(reply);
//第二个值为 channel 名称
final byte[] channelBytes = (byte[]) reply.get(1);
//转化为 string
final String channel = getString(channelBytes);
//调用事件 (onSubscribe,onPSubscribe 子类可以重写)
if (isSubscribe) {
onSubscribe(channel);
} else {
onPSubscribe(channel);
}
return true;
}
return false;
}
public abstract void onSubscribe(final String channel);
public abstract void onPSubscribe(final String channelPatterns);
在 MyPubSub
中重写上述两个方法。
@Override
public void onSubscribe(String channel) {
System.out.println("订阅了:"+channel);
}
@Override
public void onPSubscribe(String channelPatterns) {
System.out.println("订阅了:"+channelPatterns);
}
这样,我们就能够收到回调消息了。
订阅了:news.*
接收到消息同理:
@Override
public void onMessage(String channel, String message) {
System.out.println(channel + " 收到了消息:"+message);
}
channel1 收到了消息:welcome to NBA.
不过这里需要注意的一点是,在普通订阅的消息中只有【MESSAGE,CHANNEL,CONTENT】三个值,而模式匹配的订阅消息中,有【PMESSAGE,PATTERN,CHANNEL,CONTENT】四个值,其中就多了一个 PATTERN
,也就是上文中的news.*
,所以稍微做一下区分就可以了
private boolean handleMessage(byte[] resp, List<Object> reply) {
boolean isMessage = Arrays.equals(MESSAGE.raw, resp);
boolean isPMessage = Arrays.equals(PMESSAGE.raw, resp);
if (isMessage || isPMessage) {
final byte[] secondBytes = (byte[]) reply.get(1);
final byte[] thirdBytes = (byte[]) reply.get(2);
final String second = getString(secondBytes);
final String third = getString(thirdBytes);
if (isMessage) {
onMessage(second, third);
} else {
final byte[] messageBytes = (byte[]) reply.get(3);
final String message = getString(messageBytes);
onPMessage(second, third, message);
}
return true;
}
调用示例
news.country(news.*)收到了消息:this is china news
总结
本文简单的对Redis
的Pub/Sub
模式做了介绍,并且在客户端中做了相应的处理。当然其中也是大量参考了Jedis
源码。本文就到这里啦,88
源码链接
Redis的Pub/Sub客户端实现的更多相关文章
- 【转】 使用Redis的Pub/Sub来实现类似于JMS的消息持久化
http://blog.csdn.net/canot/article/details/52040415 关于个人对Redis提供的Pub/Sub机制的认识在上一篇博客中涉及到了,也提到了关于如何避免R ...
- Redis的Pub/Sub机制存在的问题以及解决方案
Redis的Pub/Sub机制使用非常简单的方式实现了观察者模式,但是在使用过程中我们发现,它仅仅是实现了发布订阅机制,但是很多的场景没有考虑到.例如一下的几种场景: 1.数据可靠性无法保证 一个re ...
- Redis实战——Redis的pub/Sub(订阅与发布)在java中的实现
借鉴:https://blog.csdn.net/canot/article/details/51938955 1.什么是pub/sub Pub/Sub功能(means Publish, Subscr ...
- 带你100% 地了解 Redis 6.0 的客户端缓存
近日 Redis 6.0.0 GA 版本发布,这是 Redis 历史上最大的一次版本更新,包括了客户端缓存 (Client side caching).ACL.Threaded I/O 和 Redis ...
- redis的Pub/Sub
redis的Pub/Sub机制类似于广播架构,Subscriber相当于收音机,可以收听多个channel(频道),Publisher(电台)可以在channel中发布信息. 命令介绍 PUBLISH ...
- Redis系列(三)-Redis发布订阅及客户端编程
阅读目录 发布订阅模型 Redis中的发布订阅 客户端编程示例 0.3版本Hredis 发布订阅模型 在应用级其作用是为了减少依赖关系,通常也叫观察者模式.主要是把耦合点单独抽离出来作为第三方,隔离易 ...
- redis的pub/sub命令
Redis 发布订阅 Redis 发布订阅(pub/sub)是一种消息通信模式:发送者(pub)发送消息,订阅者(sub)接收消息. Redis 客户端可以订阅任意数量的频道. 下图展示了频道 cha ...
- redis的Pub/Sub功能
Pub/Sub功能(即Publish,Subscribe)意思是发布及订阅功能.简单的理解就像我们订阅blog一样,不同的是,这里的客户端与server端采用长连接建立推送机制,一个客户端发布消息,可 ...
- Redis常用操作及客户端工具
修改redis密码 打开redis.windows.conf,找到requirepass 设置密码重启服务即可 将redis安装为windows服务,批处理如下: redis-server.exe ...
随机推荐
- mybatis之使用注解
注解 使用对象 相对应的 XML 描述 @CacheNamespace 类 <cache> 为给定的命名空间(比如类)配置缓存.属性有:implemetation, eviction, f ...
- mysql 导入时报错:Got a packet bigger than‘max_allowed_packet’bytes
原因是max_allowed_packet 值设置过小. 网上粘贴一段定义: max_allowed_packet:指代mysql服务器端和客户端在一次传送数据包的过程当中数据包的大小这个是定义mys ...
- Hive安装与应用过程
1. 参考说明 参考文档: https://cwiki.apache.org/confluence/display/Hive/GettingStarted 2. 安装环境说明 2.1. 环境说明 ...
- android 自定义控件之ViewGroup生命周期执行步骤
前言 了解ViewGroup的生命周期的执行步骤对于自己自定义ViewGroup的时候十分重要,清楚了整个流程才能对ViewGroup有更深的理解.本文从个人的总结,来阐述一下执行的顺序.执行说明 首 ...
- Claims-based认证解析
Claims-based认证相关的两个重要的类ClaimsIdentity以及ClaimsPrincipal解析 ClaimsIdentity以及ClaimsPrincipal是.NET下Claims ...
- apache ftp server的简单入门(properties验证)
Apache FTPServer:(开源) Apache FTPServer是一个100%纯Java的FTP服务器. 它的设计是基于现有的开放式协议的完整和便携式FTP服务器引擎解决方案.FTPSer ...
- DELETE语句总结
一.基本语句 1.SQL DELETE 语法 DELETE FROM table_name WHERE condition; 请注意 删除表格中的记录时要小心!注意SQL DELETE 语句中的 WH ...
- PHP接收IOS post过来的json数据无法解析的问题
在本地环境下运行解析OK 换到线上的环境解析失败 开始怀疑各种编码问题,解决均无效. 查看phpinfo 发现magic_quotes_gpc =on 终于找到问题所在,更改php.ini文件 mag ...
- MySQL->>innodb_autoinc_lock_mode参数控制auto_increment 插入数据时相关锁的模式
转自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/15498/viewspace-2141640/ ---------------------------------- ...
- C语言占位符
占位符就是先占住一个固定的位置,等着你再往里面添加内容的符号,广泛用于计算机中各类文档的编辑. 格式占位符(%)是在C/C++语言中格式输入函数,如 scanf.printf 等函数中使用.其意义就是 ...