ActiveMQ 持久订阅者,执行结果与初衷相违背,验证离线订阅者无效,问题解决
导读
最新在接触ActiveMQ,里面有个持久订阅者模块,功能是怎么样也演示不出来效果。配置参数比较简单(配置没几个参数),消费者第一次运行时,需要指定ClientID(此时Broker已经记录离线订阅者信息),在启动提供者,此时消息队列存在一条记录,然后在启动消费者,但是怎么样也获取不到消息,阿西吧~~~什么鬼,百度上一大堆,都是这样步骤,消费者端,指定以下ClientID就好了,可,想要的效果死活不出来。。。。。。
采坑之路
废话不多说,先上代码,后面再分析
消费者端代码
public void testTopicConsumer2() throws Exception {
//第一步:创建ConnectionFactory
String brokerURL = "tcp://192.168.31.215:61616";
ConnectionFactory connectionFactory = new ActiveMQConnectionFactory(brokerURL);
//第二步:通过工厂,创建Connection
Connection connection = connectionFactory.createConnection();
//设置持久订阅的客户端ID
String clientId = "10086";
connection.setClientID(clientId);
//第三步:打开链接
connection.start();
//第四步:通过Connection创建session
Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
//第五步:通过session创建Consumer
Topic topic = session.createTopic("cyb-topic");
//创建持久订阅的消费者客户端
//第一个参数是指定Topic
//第二个参数是自定义的ClientId
MessageConsumer consumer = session.createDurableSubscriber(topic, clientId);
consumer.setMessageListener(new MessageListener() {
@Override
public void onMessage(Message message) {
//第七步:处理信息
if (message instanceof TextMessage){
TextMessage tm=(TextMessage)message;
try{
System.out.println(tm.getText());
}
catch (Exception e){
e.printStackTrace();
}
}
}
});
//session.commit();
//第八步:关闭资源
consumer.close();
session.close();
connection.close();
}

只需要制定ClientID和创建持久客户端即可
提供者端代码
public void testTopicProducer() throws Exception {
Connection connection = null;
MessageProducer producer = null;
Session session = null;
try {
//第一步:创建ConnectionFactory,用于连接broker
String brokerURL = "tcp://192.168.31.215:61616";
ConnectionFactory connectionFactory = new ActiveMQConnectionFactory(brokerURL);
//设置
//((ActiveMQConnectionFactory) connectionFactory).setProducerWindowSize(1000);
//第二步:通过工厂,创建Connection
connection = connectionFactory.createConnection();
//第三步:连接启动
connection.start();
//第四步:通过连接获取session会话
//第一个参数:是否启用ActiveMQ事务,如果为true,第二个参数无用
//第二个参数:应答模式,AUTO_ACKNOWLEDGE为自动应答
session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
//第五步:通过session创建destination,两种目的地:Queue、Topic
//参数:消息队列的名称,在后台管理系统中可以看到
Topic topic = session.createTopic("cyb-topic");
//第六步:通过session创建MessageProducer
producer = session.createProducer(topic);
//producer.setDeliveryMode(DeliveryMode.PERSISTENT);
//第七步:创建Message
//方式一
//TextMessage message=new ActiveMQTextMessage();
//message.setText("queue test");
//方式二
TextMessage message1 = session.createTextMessage("topic->博客园地址:https://www.cnblogs.com/chenyanbin/");
//第八步:通过producer发送消息
producer.send(message1);
//session.commit();
} catch (Exception e) {
e.printStackTrace();
} finally {
//第九步:关闭资源
producer.close();
session.close();
connection.close();
}
}
验证离线订阅者功能
失败的验证

正确的验证方式
首先明确一点,上面的代码是没有一点问题的。为了节省时间,验证步骤和上面的差不多,不启动前两步了,直接启动第三步,也就是:
- 先启动消费者(记录持久订阅者ClientID);
- 在启动提供者;
- 启动消费者(在下面加个死循环)

问题剖析

第一次运行消费者时,此时Broker已经记录订阅者ClientID,然后程序一闪而过,进入到蓝色框中的,离线订阅者中,然后在执行提供者,此时,Topic中,已经入队一次,再次运行消费者时,运行是异步获取的,运行一闪而过(鄙人猜测,可能是ActiveMQ机制问题,内部逻辑大概是,先遍历非持久订阅者,然后在查看持久订阅者,问题出在,程序执行太快,还没到查看持久订阅者时,程序就执行完了,所以第二次执行消费者时,加了个死循环,不停监听队列消息,具体ActiveMQ底层代码没看过,有兴趣的可以研究下,底层代码找到相应位置后,记得告诉我哦~~~)

这个小问题,捣鼓一下午,百度上也说,就这2步骤配置即可,运行结果与初衷相违背,大半夜的都打算洗洗睡了,头脑风暴想出来这个方法,在下面写个死循环,不停监听队列消息,这才有了这篇博客,好啦...好啦,时间不早了,马上都快凌晨1点钟了,明个还得上班,洗洗睡了zZZZZZZZZZ
ActiveMQ 持久订阅者,执行结果与初衷相违背,验证离线订阅者无效,问题解决的更多相关文章
- Apache ActiveMQ Fileserver远程代码执行漏洞
扫端口的时候遇到8161端口,输入admin/admin,成功登陆,之前就看到过相关文章,PUT了一句话上去,但是没有什么效果,于是本地搭建了一个环境,记录一下测试过程. 环境搭建: ActiveMQ ...
- 【漏洞预警】Apache ActiveMQ Fileserver远程代码执行漏洞(CVE-2016-3088)
漏洞编码:CVE-2016-3088 实验环境:Linux Apache ActiveMQ版本号:Apache ActiveMQ 5.7.0 ----------------------------- ...
- SQLServer2005:在执行批处理时出现错误。错误消息为: 目录名无效
删除数据时忘了想delete删除的话会记录日志,更何况是我删除百万条数据,结果还没删完服务器内存就占慢了,一切数据都进不来了,估计这种情况导致我的数据库有问题了,右键打开表提示:目录名无效,执行SQL ...
- sql server在执行批处理时出现错误。错误消息为: 目录名无效
今天在客户服务器上的sql server上执行脚本,报错提示“在执行批处理时出现错误.错误消息为:目录名无效”,第一反应就是客户是不是在服务器装了360,因为之前有类似问题,360把数据库的文件给隔离 ...
- 微软BI 之SSRS 系列 - 报表邮件订阅中 SMTP 服务器匿名访问与 Windows验证, 以及如何成功订阅报表的实例
这篇文章源于在上一篇博文中有园友提出订阅 SSRS 报表时的一个问题, 于是就好好总结了一下,把有关 SSRS 报表订阅的要点和容易出现问题的地方写出来,希望对大家有所帮助! 参看上一篇博文 - S ...
- 下载一个vue项目执行npm install 后运行项目npm run dev后出错 - 问题解决
在SVN上拉下来一个vue项目,上面没有提交项目里面的node_modules文件夹,所以要自己执行 npm install 安装,但安装完后运行项目后却报错了: $ npm run dev > ...
- EF执行存储工程报错 String[4]: Size 属性具有无效大小值 0。
EF中执行存储过程报错 String[4]: Size 属性具有无效大小值 0 排查后是如下问题所致,给定的参数没有设定大小(加入红框内的就可以了) private string GetCode(MC ...
- 2016/05/13 thinkphp 3.2.2 ① 数据删除及执行原生sql语句 ②表单验证
[数据删除及执行原生sql语句] delete() 返回受影响的记录条数 $goods -> delete(30); 删除主键值等于30的记录信息 $goods -> delete( ...
- loadrunner执行场景时报Error -27040: Data Format Extension: Init: Internal error问题解决
[问题描述] 在loadrunner控制台执行场景时,所有用户均Failed,查看errors,错误原因如下: Error -27040: Data Format Extension: Init: I ...
随机推荐
- 【DataBase】更改root根用户密码 和 SQLyog安装
更改root根用户密码 和 SQLyog安装 无密码登录MySQL mysql -u root -p 修改密码与更新加密规则 ALTER USER 'root'@'localhost' IDENTIF ...
- 如何教零基础的人认识Python
前言 文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. 作者: 编程派 PS:如有需要Python学习资料的小伙伴可以加点击下方链接 ...
- F - Pearls HDU - 1300
简单dp. 题目大意:有n种珍珠,这n种珍珠有不同的需求量,不同的价格,价格越高,质量越高,在购买每一种珍珠时,都需要在原来的基础上多买10个.也就是说如果需要买x种珍珠,那就要付x+10个的钱.每一 ...
- 杭电1080 J - Human Gene Functions
题目大意: 两个字符串,可以再中间任何插入空格,然后让这两个串匹配,字符与字符之间的匹配有各自的分数,求最大分数 最长公共子序列模型. dp[i][j]表示当考虑吧串1的第i个字符和串2的第j个字符时 ...
- tensorflow版线性回归
import os os.environ['TF_CPP_MIN_LOG_LEVEL'] = '2' import tensorflow as tf def linearregression(): X ...
- ubuntu下载速度慢的解决办法--修改下载源
操作:https://blog.csdn.net/qq_24326765/article/details/81916222 推荐源:https://blog.csdn.net/qq_36328643/ ...
- docker 搭建一个wordpress 博客系统(4)
安装lnmp ()下载镜像 [root@server ~]# docker pull mysql:latest #下载mysql镜像 [root@server ~]# docker pull rich ...
- udp协议与tcp协议
TCP协议与UDP协议支持的应用协议 TCP支持的应用协议主要有:Telnet.FTP.SMTP等: UDP支持的应用层协议主要有:NFS(网络文件系统).SNMP(简单网络管理协议).DNS(主域名 ...
- ubuntu server 18.04 网络配置
从17.10开始放弃在/etc/network/interfaces里固定IP的配置 配置文件是:/etc/netplan/50-cloud-init.yaml .用缩进来表示层级关系 冒号之后要有个 ...
- vs code 打开文件时,取消文件目录的自动定位跟踪
文件-->首选项-->设置-->在搜索栏中搜索:explorer.autoReveal; 去掉勾选即可.