博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
ActiveMQ学习-Network connectors
阅读量:6453 次
发布时间:2019-06-23

本文共 13657 字,大约阅读时间需要 45 分钟。

hot3.png

本文为network connectors的static connector学习笔记。

 

Network connectors

 

broker网络能够创建多个相互连接的ActiveMq实例组成的簇,以应对更加复杂的消息场景。Network connectors提供了broker之间的通信。

默认情况下,network connector是单向通道,它只会把收到的消息投递给与之建立连接的另一个broker。这通常称为forwarding bridge。ActiveMQ也支持双向通道,即duplex connector。下图是一个包含了这两者的复杂网络。
Network connector的XML配置如下:

一个重要的概念-discovery
discovery:是一个检测远程broker服务的进程。client通常需要感知所有的broker。broker,需要感知其他存在的broker,以建立broker的网络。
当我们想配置一个broker网络时,首要问题是:我们知道每个broker的准确地址吗?如果是,可以以静态的方式配置,将客户端连接到提前定义好的broker URI,这在你想完全控制所有资源的生产环境中比较常见。
如果客户端以及broker相互不知道彼此的地址,那么必须使用一种discovery机制来发现已有的broker。这种设置在开发环境下比较常见,易于配置和维护。
 

 

static network

概念介绍

只要我们知道了想要使用的broker的地址,就可以使用static配置方式。

 

Static connector

 

用来创建网络中多个broker的静态配置。协议使用组合URI,即URI中包含其他URI。格式如下:

static:(uri1,uri2,uri3,...) ?key=value
XML中配置示例:

 

程序实例

为了更好的理解,可以通过一个发布者-订阅者的例子来进行说明。(demo来自ActiveMQ in action上的例子)

这个例子使用下图所示的broker拓扑结构:
BrokerA与brokerB单向相连,当生产者把消息发送给brokerA时,他们会被投递给有订阅需求的broker。这个时候,会被brokerA投递给brokerB。
详细代码如下。
brokerB配置(brokerB.xml):

brokerA配置(brokerA.xml):

 

 

 

消息生产者(Publisher.java):

/**  * XXX.com Inc.  * Copyright (c) 2004-2015 All Rights Reserved.  */  package com.test.SpringTest.activemqinaction.ch4;    import java.util.Hashtable;  import java.util.Map;    import javax.jms.Connection;  import javax.jms.ConnectionFactory;  import javax.jms.Destination;  import javax.jms.JMSException;  import javax.jms.MapMessage;  import javax.jms.Message;  import javax.jms.MessageProducer;  import javax.jms.Session;    import org.apache.activemq.ActiveMQConnectionFactory;  import org.apache.activemq.command.ActiveMQMapMessage;    /**  * 消息产生者  *   * @author jiangnan  * @version $Id: Publisher.java, v 0.1 2015年7月4日 下午4:48:48 jiangnan Exp $  */  public class Publisher {        protected int                                MAX_DELTA_PERCENT = 1;      protected Map
LAST_PRICES = new Hashtable
(); protected static int count = 10; protected static int total; protected static String brokerURL = "tcp://localhost:61616"; protected static transient ConnectionFactory factory; protected transient Connection connection; protected transient Session session; protected transient MessageProducer producer; public Publisher() throws JMSException { factory = new ActiveMQConnectionFactory(brokerURL); connection = factory.createConnection(); connection.start(); session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE); producer = session.createProducer(null); } public void close() throws JMSException { if (connection != null) { connection.close(); } } public static void main(String[] args) throws JMSException { String[] topics = { "topic1", "topic2" }; Publisher publisher = new Publisher(); while (total < 1000) { for (int i = 0; i < count; i++) { publisher.sendMessage(topics); } total += count; System.out.println("Published '" + count + "' of '" + total + "' price messages"); try { Thread.sleep(1000); } catch (InterruptedException x) { } } publisher.close(); } protected void sendMessage(String[] stocks) throws JMSException { int idx = 0; while (true) { idx = (int) Math.round(stocks.length * Math.random()); if (idx < stocks.length) { break; } } String stock = stocks[idx]; Destination destination = session.createTopic("STOCKS." + stock); Message message = createStockMessage(stock, session); System.out.println("Sending: " + ((ActiveMQMapMessage) message).getContentMap() + " on destination: " + destination); producer.send(destination, message); } protected Message createStockMessage(String stock, Session session) throws JMSException { Double value = LAST_PRICES.get(stock); if (value == null) { value = new Double(Math.random() * 100); } // lets mutate the value by some percentage double oldPrice = value.doubleValue(); value = new Double(mutatePrice(oldPrice)); LAST_PRICES.put(stock, value); double price = value.doubleValue(); double offer = price * 1.001; boolean up = (price > oldPrice); MapMessage message = session.createMapMessage(); message.setStringProperty("stock", stock);//设置消息的属性 message.setString("stock", stock); message.setDouble("price", price); message.setDouble("offer", offer); message.setBoolean("up", up); return message; } protected double mutatePrice(double price) { double percentChange = (2 * Math.random() * MAX_DELTA_PERCENT) - MAX_DELTA_PERCENT; return price * (100 + percentChange) / 100; } }

 

消息订阅者(Consumer.java):

/**  * XXX.com Inc.  * Copyright (c) 2004-2015 All Rights Reserved.  */  package com.test.SpringTest.activemqinaction.ch4;    import javax.jms.Connection;  import javax.jms.ConnectionFactory;  import javax.jms.Destination;  import javax.jms.JMSException;  import javax.jms.MessageConsumer;  import javax.jms.Session;    import org.apache.activemq.ActiveMQConnectionFactory;    /**  * 消息消费者  *   * @author jiangnan  * @version $Id: Consumer.java, v 0.1 2015年7月4日 下午4:37:48 jiangnan Exp $  */  public class Consumer {        private static String                      brokerURL = "tcp://localhost:61617";      private static transient ConnectionFactory factory;      private transient Connection               connection;      private transient Session                  session;        public Consumer() throws JMSException {          factory = new ActiveMQConnectionFactory(brokerURL);          connection = factory.createConnection();          connection.start();          session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);      }        public void close() throws JMSException {          if (connection != null) {              connection.close();          }      }        public static void main(String[] args) throws JMSException {          String[] topics = { "topic1", "topic2" };          Consumer consumer = new Consumer();          for (String stock : topics) {              Destination destination = consumer.getSession().createTopic("STOCKS." + stock);              //只接收部分消息的选择器              String selector = "stock = 'topic1'";              MessageConsumer messageConsumer = consumer.getSession().createConsumer(destination,                  selector);              messageConsumer.setMessageListener(new Listener());          }      }        public Session getSession() {          return session;      }    }

消息监听器(Listener.java):

/**  * XXX.com Inc.  * Copyright (c) 2004-2015 All Rights Reserved.  */  package com.test.SpringTest.activemqinaction.ch4;    import java.text.DecimalFormat;    import javax.jms.MapMessage;  import javax.jms.Message;  import javax.jms.MessageListener;    /**  * 消息监听器  *   * @author jiangnan  * @version $Id: Listener.java, v 0.1 2015年7月4日 下午4:40:25 jiangnan Exp $  */  public class Listener implements MessageListener {        public void onMessage(Message message) {          try {              MapMessage map = (MapMessage) message;              String stock = map.getString("stock");              double price = map.getDouble("price");              double offer = map.getDouble("offer");              boolean up = map.getBoolean("up");              DecimalFormat df = new DecimalFormat("#,###,###,##0.00");              System.out.println(stock + "\t" + df.format(price) + "\t" + df.format(offer) + "\t"                                 + (up ? "up" : "down"));          } catch (Exception e) {              e.printStackTrace();          }      }    }

运行过程:
先启动brokerB:
D:\apache-activemq-5.8.0\bin>activemq xbean:file:D:/code/test/SpringTest/src/main/resource/META-INF/spring/activemqinaction/ch4/brokerB.xml
 

 

再启动brokerA:

D:\apache-activemq-5.8.0\bin>activemq xbean:file:D:/code/test/SpringTest/src/main/resource/META-INF/spring/activemqinaction/ch4/brokerA.xml

可以看到brokerA和brokerB建立连接成功。

然后启动消费者和生产者。在控制台可以观察到消息发送和接收的日志。

 

Static protocol的使用场景

考虑这样一种场景,多个远程客户端与本地的一个broker建立连接。考虑到每个远程区域的客户端数量,与本地broker关联的连接数会很多。这会给网络带来不必要的负担。为了减小连接数,可以在每个远程区域设置一个broker,然后在远程broker和本地broker之间建立静态链接。这不仅会减小网络连接数,也会提高客户端工作效率。同时也会减少延时,降低等待客户端的时间。

 

Failover connector

概念介绍

在之前的例子中,客户端仅仅连接到一个特定的broker。如果连接失败或中断,怎么办?有两个选择:客户端会消亡,或者是重新连接到这个broker或者其他broker然后恢复工作。failover可以实现自动重连。有两种方式可以为客户端提供可以连接的broker,一是提供一个静态列表,二是使用动态发现机制。

静态列表配置格式如下:

failover:(uri1,...,uriN)?key=value
或者
failover:uri1,...,uriN
默认情况下,会随机挑选可使用的connector。如果连接失败,会挑选另一个URI尝试建立连接。默认配置实现了重连延迟逻辑:第一次重试失败后延迟10ms,之后延迟时间都在前一次的时间之上加倍,直至30000ms。
使用场景
强烈推荐为所有客户端使用failover,即时客户端只会连接到一个broker。这样做的好处是,broker挂掉之后不用手动重新连接,broker恢复后客户端会自动进行重连。简单的利用ActiveMQ的这一特性可以使应用更加稳定。
 

参考资料:

 

《ActiveMQ in Action》

 

附:demo的pom.xml配置

4.0.0
com.test
SpringTest
0.0.1-SNAPSHOT
jar
SpringTest
http://maven.apache.org
UTF-8
3.1.2.RELEASE
junit
junit
4.11
test
com.sun.jdmk
jmxtools
1.2.1
org.apache.activemq
activemq-all
5.6.0
org.apache.activemq
activemq-core
5.5.0
org.apache.xbean
xbean-spring
3.7
org.springframework
spring-core
${org.springframework.version}
org.springframework
spring-expression
${org.springframework.version}
org.springframework
spring-beans
${org.springframework.version}
org.springframework
spring-aop
${org.springframework.version}
org.springframework
spring-context
${org.springframework.version}
org.springframework
spring-context-support
${org.springframework.version}
org.springframework
spring-tx
${org.springframework.version}
org.springframework
spring-jdbc
${org.springframework.version}
org.springframework
spring-orm
${org.springframework.version}
org.springframework
spring-oxm
${org.springframework.version}
org.springframework
spring-web
${org.springframework.version}
org.springframework
spring-webmvc
${org.springframework.version}
org.springframework
spring-webmvc-portlet
${org.springframework.version}
org.springframework
spring-test
${org.springframework.version}
test
org.springframework
spring-jms
3.1.1.RELEASE

 

转载于:https://my.oschina.net/u/3391025/blog/1609189

你可能感兴趣的文章
论Optimizer的工作模式ALL_ROWS&FIRST_ROWS
查看>>
生产环境高并发MySQL SQL语句优化案例
查看>>
Lync 小技巧-24-PDF 加密文件-转-Word-操作手册
查看>>
ASP.NET性能优化之分布式Session
查看>>
TaffyDB Introduction
查看>>
Piwik 1.9.1 发布,网站访问统计系统
查看>>
转载:《TypeScript 中文入门教程》 16、Symbols
查看>>
JavaScript、jQuery、HTML5、Node.js实例大全-读书笔记4
查看>>
C#技术------垃圾回收机制(GC)
查看>>
【转】eclipse -- the project was not built due to a resource exists with a different case...
查看>>
漫谈并发编程(三):共享受限资源
查看>>
【转】github如何删除一个仓库
查看>>
Linux系统编程——进程调度浅析
查看>>
大数据Lambda架构
查看>>
openCV_java 图像二值化
查看>>
状态模式
查看>>
删除CentOS / RHEL的库和配置文件(Repositories and configuraiton files)
查看>>
DJANGO变动库的一次真实手动经历
查看>>
8个基本的引导工具的网页设计师
查看>>
【下载分】C语言for循环语句PK自我活动
查看>>