最近同事在使用AMQ进行消息通信时遇到了问题:
在IE8中多个页面上访问AMQ的一个主题订阅的时候,多个页面中获得的消息其实是来自于同一个主题队列,导致页面状态逻辑错误.
查找AMQ的官方文档后得知,在5.4.2之后的版本中,AMQ的AJAX客户端可以通过指定唯一的clientId,让多个页面在一个浏览器中能够访问相同订阅的不同队列,避免了之前版本中只判断JSESSIONID导致的客户端无法区分的问题.官方文档在这里:
http://activemq.apache.org/ajax.html
但这里面实现的amq.js由于改动过大,无法与之前的接口兼容,所以并没有合并到官方主干源码中,只是以demo的形式额外存放.但实现上述功能的后端代码从5.4.2之后都被主干源码合并.
所以,单纯使用最新版的AMQ,是无法得到这个AJAX多客户端的功能的,需要自己额外进行调整.
今天上午解决了这个问题,稍稍修改了一下_amq.js文件,保证了与原接口的一致性,又增加了initClientId方法,可以方便的使用AMQ的AJAX多客户端功能了.
使用时,可以将这个_amq.js替换原activemq_web-x.x.x.jar中的文件,同时在调用页面中,进行最初uri赋值的时候,顺手调用一下amq.initClientId();就可以开启这个功能了.
刚刚进行代码测试,初步通过,功能的正确性得到了初步验证.
如果有同学遇到了相同的问题,希望能够帮忙进行测试调整.
文章系本人原创,转载请注明出处和作者
,本文原址在http://h-rain.iteye.com/blog/1387640
/**
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
// AMQ Ajax handler
// This class provides the main API for using the Ajax features of AMQ. It
// allows JMS messages to be sent and received from javascript when used
// with the org.apache.activemq.web.MessageListenerServlet
//
var amq =
{
// The URI of the MessageListenerServlet
uri: '/amq',
// Polling. Set to true (default) if waiting poll for messages is needed
poll: true,
// Poll delay. if set to positive integer, this is the time to wait in ms before
// sending the next poll after the last completes.
_pollDelay: 0,
_clientIdStr:'',
_first: true,
_pollEvent: function(first) {},
_handlers: new Array(),
_messages:0,
_messageQueue: '',
_queueMessages: 0,
//init the clientId,default auto make as Date string
initClientId:function(newId)
{
if (newId)
amq._clientIdStr="clientId="+newId;
else
amq._clientIdStr="clientId="+(new Date()).getTime().toString();
},
//using clientId,whit separtor
_makeClientIdStr:function(spChar)
{
var SC=spChar||'';
if (amq._clientIdStr=='')
return '';
return SC+amq._clientIdStr;
},
_messageHandler: function(request)
{
try
{
if (request.status == 200)
{
var response = request.responseXML.getElementsByTagName("ajax-response");
if (response != null && response.length == 1)
{
for ( var i = 0 ; i < response[0].childNodes.length ; i++ )
{
var responseElement = response[0].childNodes[i];
// only process nodes of type element.....
if ( responseElement.nodeType != 1 )
continue;
var id = responseElement.getAttribute('id');
var handler = amq._handlers[id];
if (handler!=null)
{
for (var j = 0; j < responseElement.childNodes.length; j++)
{
handler(responseElement.childNodes[j]);
}
}
}
}
}
}
catch(e)
{
alert(e);
}
},
startBatch: function()
{
amq._queueMessages++;
},
endBatch: function()
{
amq._queueMessages--;
if (amq._queueMessages==0 && amq._messages>0)
{
var body = amq._messageQueue;
amq._messageQueue='';
amq._messages=0;
amq._queueMessages++;
new Ajax.Request(amq.uri, { method: 'post', postBody: body+amq._makeClientIdStr('&'), onSuccess: amq.endBatch});
}
},
_pollHandler: function(request)
{
amq.startBatch();
try
{
amq._messageHandler(request);
amq._pollEvent(amq._first);
amq._first=false;
}
catch(e)
{
alert(e);
}
amq.endBatch();
if (amq._pollDelay>0)
setTimeout('amq._sendPoll()',amq._pollDelay);
else
amq._sendPoll();
},
_sendPoll: function(request)
{
new Ajax.Request(amq.uri, { method: 'get', parameters: amq._makeClientIdStr(),onSuccess: amq._pollHandler });
},
// Add a function that gets called on every poll response, after all received
// messages have been handled. The poll handler is past a boolean that indicates
// if this is the first poll for the page.
addPollHandler : function(func)
{
var old = amq._pollEvent;
amq._pollEvent = function(first)
{
old(first);
func(first);
}
},
// Send a JMS message to a destination (eg topic://MY.TOPIC). Message should be xml or encoded
// xml content.
sendMessage : function(destination,message)
{
amq._sendMessage(destination,message,'send');
},
// Listen on a channel or topic. handler must be a function taking a message arguement
addListener : function(id,destination,handler)
{
amq._handlers[id]=handler;
amq._sendMessage(destination,id,'listen');
},
// remove Listener from channel or topic.
removeListener : function(id,destination)
{
amq._handlers[id]=null;
amq._sendMessage(destination,id,'unlisten');
},
_sendMessage : function(destination,message,type)
{
if (amq._queueMessages>0)
{
if (amq._messages==0)
{
amq._messageQueue='destination='+destination+'&message='+message+'&type='+type;
}
else
{
amq._messageQueue+='&d'+amq._messages+'='+destination+'&m'+amq._messages+'='+message+'&t'+amq._messages+'='+type;
}
amq._messages++;
}
else
{
amq.startBatch();
new Ajax.Request(amq.uri, { method: 'post', postBody: 'destination='+destination+'&message='+message+'&type='+type+amq._makeClientIdStr('&'), onSuccess: amq.endBatch});
}
},
_startPolling : function()
{
if (amq.poll)
new Ajax.Request(amq.uri, { method: 'get', parameters: 'timeout=10'+amq._makeClientIdStr('&'), onSuccess: amq._pollHandler });
}
};
Behaviour.addLoadEvent(amq._startPolling);
function getKeyCode(ev)
{
var keyc;
if (window.event)
keyc=window.event.keyCode;
else
keyc=ev.keyCode;
return keyc;
}
分享到:
相关推荐
- **Amq_Producer_mt.cpp**:扩展了 Amq_Producer.cpp,增加了多线程支持,每个线程独立发送消息,提高消息发送速率。 - **Amq_Consumer.cpp**:消费者客户端的实现,负责接收和处理来自 ActiveMQ 服务器的消息。 ...
8. **管理工具**:ActiveMQ提供了一个Web控制台(webconsole),可以通过浏览器访问,进行管理操作,如查看和管理消息、监控性能等。 9. **安全性**:ActiveMQ支持基于角色的访问控制(RBAC)、SSL加密以及JAAS认证...
基于SpringBoot开发的ActiveMQ虚拟主题客户端,达到消费者多点消费、负载均衡和故障转移的目的。 详情可以参考这里: https://my.oschina.net/noryar/blog/1573047 https://my.oschina.net/noryar/blog/1575003
2. **发送消息**:使用Ajax,客户端可以向ActiveMQ发布消息,这通常涉及构建JMS消息对象,设置目的地(如主题或队列),然后通过Ajax请求发送到服务器。 3. **接收消息**:同样,客户端会监听服务器的响应,一旦有新...
标题中的“activemq spring 客户端配置”指的是如何在Spring框架中设置Apache ActiveMQ作为消息中间件的客户端。ActiveMQ是Apache软件基金会开发的一个开源消息代理,它实现了多种消息协议,如JMS(Java Message ...
总结来说,了解 MQTT 协议的基本原理和 ActiveMQ 的 MQTT 支持是实现 MQTT 客户端的关键。通过正确配置 ActiveMQ 代理,并利用 MQTT 客户端库,可以有效地实现在物联网设备间或应用程序间的消息通信。
1. **多客户端支持**:修改代码以支持多个消费者和生产者,模拟真实世界的并发消息收发。 2. **消息确认机制**:研究和实现ActiveMQ的确认机制,确保消息被正确处理。 3. **异常处理**:添加适当的错误处理代码,如...
标题中的"ActiveMQ 使用Ajax 收发消息实战"指出我们将探讨如何使用ActiveMQ消息中间件与Ajax技术结合,实现Web应用程序中的异步消息传递。ActiveMQ是Apache软件基金会的一个项目,提供了一个开源的消息代理,支持...
在本示例中,我们将探讨如何利用Ajax技术与ActiveMQ进行交互,实现客户端的异步消息发送和接收。 Ajax(Asynchronous JavaScript and XML)是一种在无需刷新整个网页的情况下更新部分网页内容的技术。在与ActiveMQ...
在本文中,我们将深入探讨如何使用C# WinForms来创建一个ActiveMQ服务端,并实现多客户端的通信。这将帮助初学者理解ActiveMQ在分布式系统中的作用以及如何通过C#进行实际应用。 首先,我们需要安装Apache ActiveMQ...
Topic适用于发布/订阅模式,多客户端可以订阅同一个主题并接收消息;Queue则采用点对点模式,消息只会被一个消费者接收,适合实现负载均衡。 **3. 服务端实现** 服务端是消息的生产者,它会发送消息到ActiveMQ。...
ActiveMQ是Apache软件基金会开发的一款开源、高性能的消息中间件,支持多种消息协议,如AMQP、STOMP、OpenWire等。在本场景中,我们将讨论如何利用ActiveMQ和Ajax技术来实现一个多人聊天室的功能。 首先,了解...
【标题】:“实验三 消息中间件应用开发:ActiveMQ实现单线程多队列” 在IT领域,消息中间件是一种重要的软件架构组件,它主要用于应用程序之间的异步通信,提高系统的可扩展性和解耦性。本实验主要关注的是如何...
Apache ActiveMQ 是一款开源的消息中间件,它遵循开放消息中间件协议(Open Message Broker Protocol),支持多种消息协议,如 OpenWire、STOMP、AMQP、MQTT 和 WebSocket 等。在 Java 开发环境中,ActiveMQ 提供了...
**ActiveMQ MQTT Android 客户端Demo详解** 在Android应用开发中,实时数据通信是非常重要的一环,而MQTT(Message Queuing Telemetry Transport)协议因其轻量级、低延迟和高可靠性的特点,常被用于物联网(IoT)...
最新在做ActiveMQ Web端开发时,绕了很多路找到了这个插件,小编在这里就贡献了给爱学习小伙伴用。
ActiveMQ支持多种协议,包括开放消息传递协议(Openwire)、AMQP、STOMP、MQTT和WebSockets,这使得它能够与多种编程语言和平台无缝集成。在示例程序中,你可以学习如何创建生产者和消费者,以及如何配置ActiveMQ...