activemq 的那些事1
#关于事务:
activemq 遇到的不能消息确认的问题。
Session session = connection.createSession(Boolean.FALSE, Session.AUTO_ACKNOWLEDGE);
有两种情况:
1 开启事务
当createSession第一个参数为true时,表示创建的session被标记为transactional的,确认消息就通过确认和校正来自动地处理,第二个参数应该是没用的。 此时需要我们手动 session.commit(); , 否则此session发送的消息是不会发送到activemq 中去,而只是存在于当前jvm 内存..
2 不开启事务
当createSession的第一个参数为false时,表示创建的session没有标记为transactional, 此时不能执行session.commit(), 否则报错的, 因为:
public void commit() throws JMSException {
this.checkClosed();
if (!this.getTransacted()) {
throw new IllegalStateException("Not a transacted session");
} else {
if (LOG.isDebugEnabled()) {
LOG.debug(this.getSessionId() + " Transaction Commit :" + this.transactionContext.getTransactionId());
}
this.transactionContext.commit();
}
}
此时有三种用于消息确认的选项:
**AUTO_ACKNOWLEDGE session将自动地确认收到的一则消息;
**CLIENT_ACKNOWLEDGE 客户端程序将确认收到的一则消息,调用这则消息的确认方法;
**DUPS_OK_ACKNOWLEDGE 这个选项命令session“懒散的”确认消息传递,可以想到,这将导致消息提供者传递的一些复制消息可能出错。
#关于消息持久化:
有两种方式NON_PERSISTENT、PERSISTENT
MS有两种消息传递方式。标记为NON_PERSISTENT的消息最多传递一次,而标记为PERSISTENT的消息将使用暂存后再转发的机理投递。如果一个JMS服务离线,那么持久性消息不会丢失,但是得等到这个服务恢复联机的时候才会被传递。所以默认的消息传递方式是非持久性的,虽然使用非持久性消息可能降低内存和需要的存储器,但这种传递方式只有当你不需要接收所有消息时才使用。
我自己的理解: 持久化的消息呢, activemq 服务器重启后, 消息还在, 非持久化的消息activemq 服务器重启后, 消息就不在了..
#关于消息无法消费:
消息无法消费通常有各种各样的原因, 这里说的是connection 关闭过早的问题:
下面的代码, 如果 consume();后面直接, close(); 的话, MessageListener 是消费不到的, 因为 还没来得及消费, connection 就已经关闭了啊..
有一种情况例外,: consumer.receive(TIME_IN_MILLISECONDS); 方法, 我测试的时候, 如果把 setMessageListener 改成receive, 那么, 我们是可以消费到一条消息的( 只要queue 存在未消费消息 )...
public class TestActiveMq{
private static final int SEND_NUMBER = 5;
static ConnectionFactory connectionFactory;
// Connection :JMS 客户端到JMS Provider 的连接
static Connection connection = null;
static String admin = "admin";
static String pass = "admin";
static String brokerUrl = "tcp://10.10.10.123:61616";
public static void main(String[] args) throws Exception {
init();
consume();
Thread.sleep(2000);
close();
}
public static void init() {
// ConnectionFactory :连接工厂,JMS 用它创建连接
// Session: 一个发送或接收消息的线程
// TextMessage message;
// 构造ConnectionFactory实例对象,此处采用ActiveMq的实现jar
connectionFactory = new ActiveMQConnectionFactory(
admin,
pass,
// ActiveMQConnection.DEFAULT_PASSWORD,
brokerUrl);
((ActiveMQConnectionFactory) connectionFactory).setTrustAllPackages(true);
try {
// 构造从工厂得到连接对象
connection = connectionFactory.createConnection();
// 启动
connection.start();
// 获取操作连接
} catch (Exception e) {
e.printStackTrace();
} finally {
}
}
public static void close() {
try {
if (null != connection)
connection.close();
} catch (Throwable ignore) {
}
}
/**
*
* 队列操作
*/
public static void consume() throws JMSException {
Session session = connection.createSession(false, Session.CLIENT_ACKNOWLEDGE);
String qu = null;
qu = "test-persistence";
Queue queue = session.createQueue(qu);
MessageConsumer consumer = session.createConsumer(queue);
System.out.println("consumer = " + consumer);
consumer.setMessageListener(new MessageListener() {
@Override
public void onMessage(Message message) {
System.out.println("message00 ============ " + message);
try {
message.acknowledge();
} catch (JMSException e) {
e.printStackTrace();
}
}
});
}
public static void sendMessage()
throws Exception {
Session session;
// Destination :消息的目的地;消息发送给谁.
Destination destination;
// MessageProducer:消息发送者
MessageProducer producer;
session = connection.createSession(Boolean.FALSE,
Session.AUTO_ACKNOWLEDGE);
// 获取session注意参数值xingbo.xu-queue是一个服务器的queue,须在在ActiveMq的console配置
destination = session.createQueue("test-persistence");
// 得到消息生成者【发送者】
producer = session.createProducer(destination);
// 设置不持久化,可以更改
producer.setDeliveryMode(DeliveryMode.NON_PERSISTENT);
for (int i = 1; i <= SEND_NUMBER; i++) {
TextMessage message = session
.createTextMessage("ActiveMq 发送的消息" + i);
// 发送消息到目的地方
System.out.println("发送消息:" + i);
long delay = 5 * 1000;
long period = 10 * 1000;
int repeat = 5;
// message.setLongProperty(ScheduledMessage.AMQ_SCHEDULED_DELAY, delay);
// message.setLongProperty(ScheduledMessage.AMQ_SCHEDULED_PERIOD, period);
// message.setIntProperty(ScheduledMessage.AMQ_SCHEDULED_REPEAT, repeat);
// message.setJMSExpiration(2);
// message.setJMSTimestamp(period);
// producer.setTimeToLive(0);
producer.send(message);
}
// session.commit();
}
/**
*
* 队列操作
*/
public static Object browser() throws JMSException {
Session session = connection.createSession(false, 2);
String qu = null;
qu = "test-persistence";
Queue queue = session.createQueue(qu);
QueueBrowser browser = session.createBrowser(queue);
List<String> msgList = new ArrayList<>();
Enumeration enumeration = browser.getEnumeration();
while (enumeration.hasMoreElements()) {
// ObjectMessage o = (ObjectMessage) enumeration.nextElement();
Object o = enumeration.nextElement();
System.out.println("o = " + o);
// MessageDto object = (MessageDto) o.getObject();
// System.out.println("object = " + object);
// msgList.add(object.toString());
}
return enumeration;
}
}
参考
http://riddickbryant.iteye.com/blog/441890
http://blog.sina.com.cn/s/blog_4d22b9720102uxyr.html
https://blog.csdn.net/zishan007/article/details/45037731
https://blog.csdn.net/wsyyyyy/article/details/79888793
https://blog.csdn.net/seven__________7/article/details/69317106
activemq 的那些事1的更多相关文章
- (jms)ActiveMQ 安装配置.
前言 ActiveMQ他是Apache出品的一个JMS提供者,管理会话和队列,运行在JVM下,支持多种语言,如JAVA,C++,C#,应用协议: OpenWire,Stomp REST,WS Noti ...
- Spring下ActiveMQ实战
MessageQueue是分布式的系统里经常要用到的组件,一般来说,当需要把消息跨网段.跨集群的分发出去,就可以用这个.一些典型的示例就是: 1.集群A中的消息需要发送给多个机器共享: 2.集群A中消 ...
- 工业物联网或系统集成中应用消息队列(ActiveMQ,C#的demo)的场景全面分析
1.[连载]<C#通讯(串口和网络)框架的设计与实现> 2.[开源]C#跨平台物联网通讯框架ServerSuperIO(SSIO)介绍 2.应用SuperIO(SIO)和开源跨平台物联网框 ...
- ActiveMQ的初夜
Producer Flow Control mq自己实现了Flow Control(流量控制,默认开启),在mq的版本中,4.x和5.x流量控制实现原理并不相同,前者通过 TCP Flow Contr ...
- 开源 VS 商业,消息中间件你不知道的那些事
11月23日,新炬网络中间件技术专家刘拓老师在DBA+社群中间件用户组进行了一次主题为“开源 VS 商业,消息中间件你不知道的那些事”的线上分享.小编特别整理出其中精华内容,供大家学习交流. 嘉宾简介 ...
- ActiveMQ相关背景(转)
概述 介绍中间件.MOM.JMS.ActiveMQ,及相互的关系. 中间件 由于业务的不同.技术的发展.硬件和软件的选择有所差别,导致了异构组件或应用并存的局面.要使这些异构的组件协同工作,一个有效的 ...
- JMS and ActiveMQ first lesson(转)
JMS and ActiveMQ first lesson -- jms基础概念和应用场景 2011-6-18 PM 9:30 主讲:kimmking <kimmking@163.com> ...
- 关于ActiveMQ的一点总结
ActiveMQ入门 作者:一路向北 摘要:本文主要讲述ActiveMQ的基本知识和使用方法,并简单结合spring使用ActiveMQ. 一.ActiveMQ特性和使用总览 企业消息软件从80年代起 ...
- 消息队列-ActiveMQ
1 业务需求描述 举例描述: 再警情通报的业务时通过发送消息界面可以选择 警情联络,和船情通报两种消息 发送方式可分为 一对一发送:部门对部门.个人对个人 一对多发送:部门对多部门.个人对多人 2 功 ...
随机推荐
- boot项目swagger接口调试工具默认访问路径
今天第一次接触boot项目,项目里集成了swagger接口调试工具,以前写项目的时候在swagger 文件夹里可以找到访问路径,换成boot项目以后找不到swagger文件夹了,百度了一下,boot项 ...
- mysql错误集合
一.This function has none of DETERMINISTIC, NO SQL, or READS SQL DATA in its de 错误解决办法 这是我们开启了bin-log ...
- Alpha冲刺7
前言 队名:拖鞋旅游队 组长博客:https://www.cnblogs.com/Sulumer/p/10013652.html 作业博客:https://edu.cnblogs.com/campus ...
- Linux系统-tcpdump常用抓包命令
主要语法 过滤主机/IP: tcpdump -i eth1 host 172.16.7.206 抓取所有经过网卡1,目的IP为172.16.7.206的网络数据 过滤端口: tcpdump -i e ...
- 蓝牙协议分析(3)_BLE协议栈介绍
1. 前言 通过“蓝牙协议分析(2)_协议架构”的介绍,大家对蓝牙协议栈应该有了简单的了解,但是,肯定还有“似懂非懂.欲说还休”的感觉.有这种感觉太正常了,毕竟蓝牙协议是一个历史悠久又比较庞大的协议, ...
- BULK语句 将TXT数据塞入数据库表格
SET @iSQL=N'BULK INSERT [TEST].[dbo].[TEST_Interim]'+' FROM '+quotename(@fullFileName,'''')+' WITH ( ...
- python实现算术表达式的词法语法语义分析(编译原理应用)
本学期编译原理的一个大作业,我的选题是算术表达式的词法语法语义分析,当时由于学得比较渣,只用了递归下降的方法进行了分析. 首先,用户输入算术表达式,其中算术表达式可以包含基本运算符,括号,数字,以及用 ...
- Appium·基础+项目
date:2018609 day13 一.Appium Appium是一个开源.跨平台的测试框架.可以用来测试原生以及混合的移动端应用. 1.安装操作 ①.安装Appium-Python-Client ...
- Keras学习笔记(完结)
使用Keras中文文档学习 基本概念 Keras的核心数据结构是模型,也就是一种组织网络层的方式,最主要的是序贯模型(Sequential).创建好一个模型后就可以用add()向里面添加层.模型搭建完 ...
- React Native 调用 Web3(1.x) 的正确姿势
1 创建项目 react-native init lm1 cd lm1 2 安装依赖包 yarn add node-libs-browser 3 创建 rn-cli.config.js 脚本 cons ...