`
h_rain
  • 浏览: 121641 次
  • 性别: Icon_minigender_1
  • 来自: 哈尔滨
文章分类
社区版块
存档分类
最新评论

ActiveMQ(AMQ)单浏览器AJAX多页面客户端的支持(clientId)

阅读更多

    最近同事在使用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;
}
分享到:
评论
1 楼 clamking 2014-03-28  
能给个前端调用页面的代码么?我不知道什么时候调用amq.initClientId();这个方法,amq里面没有这个方法,调用出错

相关推荐

    自己实现的 ActiveMQ 多线程客户端 包含生产消息客户端和消费者消息客户端

    - **Amq_Producer_mt.cpp**:扩展了 Amq_Producer.cpp,增加了多线程支持,每个线程独立发送消息,提高消息发送速率。 - **Amq_Consumer.cpp**:消费者客户端的实现,负责接收和处理来自 ActiveMQ 服务器的消息。 ...

    ActiveMQ客户端

    8. **管理工具**:ActiveMQ提供了一个Web控制台(webconsole),可以通过浏览器访问,进行管理操作,如查看和管理消息、监控性能等。 9. **安全性**:ActiveMQ支持基于角色的访问控制(RBAC)、SSL加密以及JAAS认证...

    基于SpringBoot开发的ActiveMQ虚拟主题客户端(包含发布者和订阅者)

    基于SpringBoot开发的ActiveMQ虚拟主题客户端,达到消费者多点消费、负载均衡和故障转移的目的。 详情可以参考这里: https://my.oschina.net/noryar/blog/1573047 https://my.oschina.net/noryar/blog/1575003

    activemq ajax方式demo

    2. **发送消息**:使用Ajax,客户端可以向ActiveMQ发布消息,这通常涉及构建JMS消息对象,设置目的地(如主题或队列),然后通过Ajax请求发送到服务器。 3. **接收消息**:同样,客户端会监听服务器的响应,一旦有新...

    activemq spring 客户端配置

    标题中的“activemq spring 客户端配置”指的是如何在Spring框架中设置Apache ActiveMQ作为消息中间件的客户端。ActiveMQ是Apache软件基金会开发的一个开源消息代理,它实现了多种消息协议,如JMS(Java Message ...

    activeMQ 推送之mqtt客户端

    总结来说,了解 MQTT 协议的基本原理和 ActiveMQ 的 MQTT 支持是实现 MQTT 客户端的关键。通过正确配置 ActiveMQ 代理,并利用 MQTT 客户端库,可以有效地实现在物联网设备间或应用程序间的消息通信。

    ActiveMQ单客户端

    1. **多客户端支持**:修改代码以支持多个消费者和生产者,模拟真实世界的并发消息收发。 2. **消息确认机制**:研究和实现ActiveMQ的确认机制,确保消息被正确处理。 3. **异常处理**:添加适当的错误处理代码,如...

    ActiveMQ 使用Ajax 收发消息实战

    标题中的"ActiveMQ 使用Ajax 收发消息实战"指出我们将探讨如何使用ActiveMQ消息中间件与Ajax技术结合,实现Web应用程序中的异步消息传递。ActiveMQ是Apache软件基金会的一个项目,提供了一个开源的消息代理,支持...

    activemq 通过ajax发送接收消息简单例子

    在本示例中,我们将探讨如何利用Ajax技术与ActiveMQ进行交互,实现客户端的异步消息发送和接收。 Ajax(Asynchronous JavaScript and XML)是一种在无需刷新整个网页的情况下更新部分网页内容的技术。在与ActiveMQ...

    ActiveMQ多客户端

    在本文中,我们将深入探讨如何使用C# WinForms来创建一个ActiveMQ服务端,并实现多客户端的通信。这将帮助初学者理解ActiveMQ在分布式系统中的作用以及如何通过C#进行实际应用。 首先,我们需要安装Apache ActiveMQ...

    activeMQ推送服务端和客户端完整案例

    Topic适用于发布/订阅模式,多客户端可以订阅同一个主题并接收消息;Queue则采用点对点模式,消息只会被一个消费者接收,适合实现负载均衡。 **3. 服务端实现** 服务端是消息的生产者,它会发送消息到ActiveMQ。...

    ActiveMQ使用Ajax实现多人聊天室

    ActiveMQ是Apache软件基金会开发的一款开源、高性能的消息中间件,支持多种消息协议,如AMQP、STOMP、OpenWire等。在本场景中,我们将讨论如何利用ActiveMQ和Ajax技术来实现一个多人聊天室的功能。 首先,了解...

    实验三 消息中间件应用开发:ActiveMQ实现单线程多队列

    【标题】:“实验三 消息中间件应用开发:ActiveMQ实现单线程多队列” 在IT领域,消息中间件是一种重要的软件架构组件,它主要用于应用程序之间的异步通信,提高系统的可扩展性和解耦性。本实验主要关注的是如何...

    activeMQ 服务端客户端 java代码

    Apache ActiveMQ 是一款开源的消息中间件,它遵循开放消息中间件协议(Open Message Broker Protocol),支持多种消息协议,如 OpenWire、STOMP、AMQP、MQTT 和 WebSocket 等。在 Java 开发环境中,ActiveMQ 提供了...

    ActiveMQ MQTT Android客户端Demo

    **ActiveMQ MQTT Android 客户端Demo详解** 在Android应用开发中,实时数据通信是非常重要的一环,而MQTT(Message Queuing Telemetry Transport)协议因其轻量级、低延迟和高可靠性的特点,常被用于物联网(IoT)...

    ActiveMQ Web开发 amq.js

    最新在做ActiveMQ Web端开发时,绕了很多路找到了这个插件,小编在这里就贡献了给爱学习小伙伴用。

    ActiveMQ / RabbitMQ 示例

    ActiveMQ支持多种协议,包括开放消息传递协议(Openwire)、AMQP、STOMP、MQTT和WebSockets,这使得它能够与多种编程语言和平台无缝集成。在示例程序中,你可以学习如何创建生产者和消费者,以及如何配置ActiveMQ...

Global site tag (gtag.js) - Google Analytics