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 功 ...
随机推荐
- 二、JAVA基本数据类型:内置数据类型,引用类型
变量的值存储在内存中,内存管理系统通过变量的类型分配存储空间,且该空间只能存储该类型数据,通过定义不同的变量,在内存中储存不同类型的数据. JAVA的两大数据类型 1. 内置数据类型 2.引用数据类型 ...
- 剑指Offer 2. 替换空格 (字符串)
题目描述 请实现一个函数,将一个字符串中的每个空格替换成“%20”.例如,当字符串为We Are Happy.则经过替换之后的字符串为We%20Are%20Happy. 题目地址 https://ww ...
- 周强201771010141《面向对象程序设计Java》第八周学习总结
一.理论知识学习部分 Java为了克服单继承的缺点,Java使用了接口,一个类可以实现一个或多个接口. 接口体中包含常量定义和方法定义,接口中只进行方法的声明,不提供方法的实现. 类似建立类的继承关系 ...
- manhattan plots in qqplot2
###manhattan plots in qqplot2library(ggplot2)setwd("~/ncbi/zm/XPCLR/")read.table("LW. ...
- 几个特殊的IP地址
1)私有地址 IP地址在全世界范围内唯一,看到这句话你可能有这样的疑问,像192.168.0.1这样的地址在许多地方都能看到,并不唯一,这是为何?Internet管理委员会规定如下地址段为私有 ...
- Linq分组查询统计
这里介绍Linq使用Group By和Count得到每个CategoryID中产品的数量,Linq使用Group By和Count得到每个CategoryID中断货产品的数量等方面. 学经常会遇到Li ...
- POJ1738 An old Stone Game
题意 Language:Default An old Stone Game Time Limit: 5000MS Memory Limit: 30000K Total Submissions: 439 ...
- oracle-组件vault
===================== lsnrctl stopshutdown immediate;emctl stop dbconsolecd $ORACLE_HOME/rdbms/libma ...
- vue企业项目搭建过程(vue-cli脚手架超详细教程 傻瓜-入门)
vue作为现在主流的前端框架,有必要学习一下. vue的官方文档还是不错的,开源中文,一个爽字形容. 如果不是实际开发需要vue-cli构建项目,那么可以在加一个爽. 然而要构建的时候发现官方文档还是 ...
- HTTPS协议加密原理解析
用 HTTP 协议,看个新闻还没有问题,但是换到更加严肃的场景中,就存在很多的安全风险.例如你要下单做一次支付,如果还是使用普通的 HTTP 协议,那你很可能会被黑客盯上. 比如,你发送一个请求,说我 ...