消息中间件解决方案JMS

1. JMS入门

1.1消息中间件

1.1.1品优购系统模块调用关系分析

咱们如今讲品优购已经完成的功能模块梳理以下:java

 

咱们已经完成了5web模块和4个服务模块。其中运营商后台的调用关系最多,用到了商家商品服务、广告内容服务、搜索服务和页面生成服务。这种模块之间的依赖也称之为耦合。而耦合越多,以后的维护工做就越困难。那么若是改善系统模块调用关系、减小模块之间的耦合呢?咱们接下来就介绍一种解决方案----消息中间件。mysql

1.1.2什么是消息中间件

消息中间件利用高效可靠的消息传递机制进行平台无关的数据交流,并基于数据通讯来进行分布式系统的集成。经过提供消息传递和消息排队模型,它能够在分布式环境下扩展进程间的通讯。对于消息中间件,常见的角色大体也就有Producer(生产者)、Consumer(消费者)linux

常见的消息中间件产品:web

1ActiveMQspring

ActiveMQ Apache出品,最流行的,能力强劲的开源消息总线。ActiveMQ 是一个彻底支持JMS1.1J2EE 1.4规范的 JMS Provider实现。咱们在本次课程中介绍 ActiveMQ的使用。sql

(2)RabbitMQ数据库

AMQP协议的领导实现,支持多种场景。淘宝的MySQL集群内部有使用它进行通信,OpenStack开源云平台的通讯组件,最早在金融行业获得运用。apache

(3)ZeroMQ浏览器

史上最快的消息队列系统服务器

(4)Kafka

Apache下的一个子项目 。特色:高吞吐,在一台普通的服务器上既能够达到10W/s的吞吐速率;彻底的分布式系统。适合处理海量数据。

1.1.3改造系统模块调用关系

 

咱们经过引入消息中间件activeMQ,使得运营商系统与搜索服务、页面生成服务解除了耦合。

 

1.2 JMS简介

1.2.1什么是JMS

JMSJava Messaging Service)是Java平台上有关面向消息中间件的技术规范,它便于消息系统中的Java应用程序进行消息交换,而且经过提供标准的产生、发送、接收消息的接口简化企业应用的开发。

       JMS自己只定义了一系列的接口规范,是一种与厂商无关的 API,用来访问消息收发系统。它相似于 JDBC(java Database Connectivity):这里,JDBC 是能够用来访问许多不一样关系数据库 API,而 JMS 则提供一样与厂商无关的访问方法,以访问消息收发服务。许多厂商目前都支持 JMS,包括 IBM MQSeriesBEAWeblogic JMS serviceProgress SonicMQ,这只是几个例子。 JMS 使您可以经过消息收发服务(有时称为消息中介程序或路由器)从一个 JMS 客户机向另外一个 JML 客户机发送消息。消息是 JMS 中的一种类型对象,由两部分组成:报头和消息主体。报头由路由信息以及有关该消息的元数据组成。消息主体则携带着应用程序的数据或有效负载。

JMS 定义了五种不一样的消息正文格式,以及调用的消息类型,容许你发送并接收以一

些不一样形式的数据,提供现有消息格式的一些级别的兼容性。

· TextMessage--一个字符串对象

· MapMessage--一套名称-值对

· ObjectMessage--一个序列化的 Java 对象

· BytesMessage--一个字节的数据流

· StreamMessage -- Java 原始值的数据流

1.2.2 JMS消息传递类型

对于消息的传递有两种类型:

一种是点对点的,即一个生产者和一个消费者一一对应;

 

另外一种是发布/ 订阅模式,即一个生产者产生消息并进行发送后,能够由多个消费者进

行接收。

    

1.3ActiveMQ下载与安装

1.3.1下载

官方网站下载:http://activemq.apache.org/

1.3.2安装(Linux

(1)apache-activemq-5.12.0-bin.tar.gz 上传至服务器

(2)解压此文件

tar  zxvf  apache-activemq-5.12.0-bin.tar.gz

(3)apache-activemq-5.12.0目录赋权

chmod 777 apache-activemq-5.12.0

(4)进入apache-activemq-5.12.0\bin目录

5)赋与执行权限

chmod 755 activemq 

--------------------------------------  知识点小贴士    --------------------------

linux 命令chmod 755的意思

chmod是Linux下设置文件权限的命令,后面的数字表示不一样用户或用户组的权限。

通常是三个数字:
第一个数字表示文件全部者的权限
第二个数字表示与文件全部者同属一个用户组的其余用户的权限
第三个数字表示其它用户组的权限。

      权限分为三种:读(r=4),写(w=2),执行(x=1) 。 综合起来还有可读可执行(rx=5=4+1)、可读可写(rw=6=4+2)、可读可写可执行(rwx=7=4+2+1)。

      因此,chmod 755 设置用户的权限为: 

1.文件全部者可读可写可执行                                    --7

2.与文件全部者同属一个用户组的其余用户可读可执行 --5 
3.其它用户组可读可执行                                          --5

1.3.3启动

 ./activemq start

出现下列提示表示成功!

 

假设服务器地址为192.168.25.135 ,打开浏览器输入地址

http://192.168.25.135:8161/ 便可进入ActiveMQ管理页面

 

点击进入管理页面

 

输入用户名和密码  均为 admin

 

进入主界面

 

点对点消息列表:

 

列表各列信息含义以下:

Number Of Pending Messages  等待消费的消息 这个是当前未出队列的数量。

Number Of Consumers  消费者 这个是消费者端的消费者数量

Messages Enqueued  进入队列的消息  进入队列的总数量,包括出队列的。

Messages Dequeued  出了队列的消息  能够理解为是消费这消费掉的数量

 

2. JMS入门小Demo

2.1点对点模式

点对点的模式主要创建在一个队列上面,当链接一个列队的时候,发送端不须要知道接收端是否正在接收,能够直接向ActiveMQ发送消息,发送的消息,将会先进入队列中,若是有接收端在监听,则会发向接收端,若是没有接收端接收,则会保存在activemq服务器,直到接收端接收消息,点对点的消息模式能够有多个发送端,多个接收端,可是一条消息,只会被一个接收端给接收到,哪一个接收端先连上ActiveMQ,则会先接收到,然后来的接收端则接收不到那条消息

2.1.1消息生产者

1)建立工程jmsDemo ,引入依赖

    <dependency>

<groupId>org.apache.activemq</groupId>

<artifactId>activemq-client</artifactId>

<version>5.13.4</version>

 </dependency>

2)建立类QueueProducer  main方法代码以下:

//1.建立链接工厂

ConnectionFactory connectionFactory=new ActiveMQConnectionFactory("tcp://192.168.25.135:61616");

//2.获取链接

Connection connection = connectionFactory.createConnection();

//3.启动链接

connection.start();

//4.获取session  (参数1:是否启动事务,参数2:消息确认模式)

Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);

//5.建立队列对象

Queue queue = session.createQueue("test-queue");

//6.建立消息生产者

MessageProducer producer = session.createProducer(queue);

//7.建立消息

TextMessage textMessage = session.createTextMessage("欢迎来到神奇的品优购世界");

//8.发送消息

producer.send(textMessage);

//9.关闭资源

producer.close();

session.close();

connection.close();

上述代码中第4步建立session  的两个参数:

1个参数 是否使用事务

2个参数 消息的确认模式

  • AUTO_ACKNOWLEDGE = 1    自动确认
  • CLIENT_ACKNOWLEDGE = 2    客户端手动确认   
  • DUPS_OK_ACKNOWLEDGE = 3    自动批量确认
  • SESSION_TRANSACTED = 0    事务提交并确认

运行后经过ActiveMQ管理界面查询

 

2.1.2消息消费者

建立类QueueConsumer ,main方法代码以下:

//1.建立链接工厂

ConnectionFactory connectionFactory=new ActiveMQConnectionFactory("tcp://192.168.25.135:61616");

//2.获取链接

Connection connection = connectionFactory.createConnection();

//3.启动链接

connection.start();

//4.获取session  (参数1:是否启动事务,参数2:消息确认模式)

Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);

//5.建立队列对象

Queue queue = session.createQueue("test-queue");

//6.建立消息消费

MessageConsumer consumer = session.createConsumer(queue);

 

//7.监听消息

consumer.setMessageListener(new MessageListener() {

public void onMessage(Message message) {

TextMessage textMessage=(TextMessage)message;

try {

System.out.println("接收到消息:"+textMessage.getText());

} catch (JMSException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

});

//8.等待键盘输入

System.in.read();

//9.关闭资源

consumer.close();

session.close();

connection.close();

执行后看到控制台输出

 

2.1.3运行测试

同时开启2个以上的消费者,再次运行生产者,观察每一个消费者控制台的输出,会发现只有一个消费者会接收到消息。

2.2 发布/订阅模式

2.2.1消息生产者

建立类TopicProducer ,main方法代码以下:

//1.建立链接工厂

ConnectionFactory connectionFactory=new ActiveMQConnectionFactory("tcp://192.168.25.135:61616");

//2.获取链接

Connection connection = connectionFactory.createConnection();

//3.启动链接

connection.start();

//4.获取session  (参数1:是否启动事务,参数2:消息确认模式)

Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);

//5.建立主题对象

Topic topic = session.createTopic("test-topic");

//6.建立消息生产者

MessageProducer producer = session.createProducer(topic);

//7.建立消息

TextMessage textMessage = session.createTextMessage("欢迎来到神奇的品优购世界");

//8.发送消息

producer.send(textMessage);

//9.关闭资源

producer.close();

session.close();

connection.close();

运行效果以下:

 

2.2.2消息消费者

建立类TopicConsumer ,main方法代码以下:

//1.建立链接工厂

ConnectionFactory connectionFactory=new ActiveMQConnectionFactory("tcp://192.168.25.135:61616");

//2.获取链接

Connection connection = connectionFactory.createConnection();

//3.启动链接

connection.start();

//4.获取session  (参数1:是否启动事务,参数2:消息确认模式)

Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);

//5.建立主题对象

//Queue queue = session.createQueue("test-queue");

Topic topic = session.createTopic("test-topic");

//6.建立消息消费

MessageConsumer consumer = session.createConsumer(topic);

 

//7.监听消息

consumer.setMessageListener(new MessageListener() {

public void onMessage(Message message) {

TextMessage textMessage=(TextMessage)message;

try {

System.out.println("接收到消息:"+textMessage.getText());

} catch (JMSException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

});

//8.等待键盘输入

System.in.read();

//9.关闭资源

consumer.close();

session.close();

connection.close();

2.2.3运行测试

同时开启2个以上的消费者,再次运行生产者,观察每一个消费者控制台的输出,会发现每一个消费者会接收到消息。

 

3.Spring整合JMS

3.1 点对点模式

3.1.1消息生产者

(1)建立工程springjms_producer,在POM文件中引入SpringJms activeMQ以及单元测试相关依赖  

2)在src/main/resources下建立spring配置文件applicationContext-jms-producer.xml

<context:component-scan base-package="cn.itcast.demo"></context:component-scan>        

    <!-- 真正能够产生Connection的ConnectionFactory,由对应的 JMS服务厂商提供-->  

<bean id="targetConnectionFactory" class="org.apache.activemq.ActiveMQConnectionFactory">  

    <property name="brokerURL" value="tcp://192.168.25.135:61616"/>  

</bean>          

    <!-- Spring用于管理真正的ConnectionFactory的ConnectionFactory -->  

<bean id="connectionFactory" class="org.springframework.jms.connection.SingleConnectionFactory">  

<!-- 目标ConnectionFactory对应真实的能够产生JMS Connection的ConnectionFactory -->  

    <property name="targetConnectionFactory" ref="targetConnectionFactory"/>  

</bean>      

    <!-- Spring提供的JMS工具类,它能够进行消息发送、接收等 -->  

<bean id="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">  

    <!-- 这个connectionFactory对应的是咱们定义的Spring提供的那个ConnectionFactory对象 -->  

    <property name="connectionFactory" ref="connectionFactory"/>  

</bean>      

    <!--这个是队列目的地,点对点的  文本信息-->  

<bean id="queueTextDestination" class="org.apache.activemq.command.ActiveMQQueue">  

    <constructor-arg value="queue_text"/>  

</bean>    

3)在cn.itcast.demo包下建立消息生产者类

@Component

public class QueueProducer {

 

@Autowired

private JmsTemplate jmsTemplate;

 

@Autowired

private Destination queueTextDestination;

 

/**

 * 发送文本消息

 * @param text

 */

public void sendTextMessage(final String text){

jmsTemplate.send(queueTextDestination, new MessageCreator() {

public Message createMessage(Session session) throws JMSException {

return session.createTextMessage(text);

}

});

}

}

4)单元测试

src/test/java建立测试类

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(locations="classpath:applicationContext-jms-producer.xml")

public class TestQueue {

 

@Autowired

private QueueProducer queueProducer;

 

@Test

public void testSend(){

queueProducer.sendTextMessage("SpringJms-点对点");

}

}

3.1.2消息消费者

(1)建立工程springjms_consumer,在POM文件中引入依赖 (同上一个工程)

2)建立配置文件 applicationContext-jms-consumer-queue.xml

    <!-- 真正能够产生Connection的ConnectionFactory,由对应的 JMS服务厂商提供-->  

<bean id="targetConnectionFactory" class="org.apache.activemq.ActiveMQConnectionFactory">  

    <property name="brokerURL" value="tcp://192.168.25.135:61616"/>  

</bean>    

    <!-- Spring用于管理真正的ConnectionFactory的ConnectionFactory -->  

<bean id="connectionFactory" class="org.springframework.jms.connection.SingleConnectionFactory">  

<!-- 目标ConnectionFactory对应真实的能够产生JMS Connection的ConnectionFactory -->  

    <property name="targetConnectionFactory" ref="targetConnectionFactory"/>  

</bean>  

    <!--这个是队列目的地,点对点的  文本信息-->  

<bean id="queueTextDestination" class="org.apache.activemq.command.ActiveMQQueue">  

    <constructor-arg value="queue_text"/>  

</bean>    

<!-- 个人监听类 -->

<bean id="myMessageListener" class="cn.itcast.demo.MyMessageListener"></bean>

<!-- 消息监听容器 -->

<bean class="org.springframework.jms.listener.DefaultMessageListenerContainer">

<property name="connectionFactory" ref="connectionFactory" />

<property name="destination" ref="queueTextDestination" />

<property name="messageListener" ref="myMessageListener" />

</bean>

3)编写监听类

public class MyMessageListener implements MessageListener {

public void onMessage(Message message) {

TextMessage textMessage=(TextMessage)message;

try {

System.out.println("接收到消息:"+textMessage.getText());

} catch (JMSException e) {

e.printStackTrace();

}

}

}

4)建立测试类

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(locations="classpath:applicationContext-jms-consumer-queue.xml")

public class TestQueue {

@Test

public void testQueue(){

try {

System.in.read();

} catch (IOException e) {

e.printStackTrace();

}

}

}

3.2发布/订阅模式

3.2.1消息生产者

1)在工程springjms_producerapplicationContext-jms-producer.xml增长配置

<!--这个是订阅模式  文本信息-->  

<bean id="topicTextDestination" class="org.apache.activemq.command.ActiveMQTopic">  

    <constructor-arg value="topic_text"/>  

</bean>  

2)建立生产者类

@Component

public class TopicProducer {

@Autowired

private JmsTemplate jmsTemplate;

 

@Autowired

private Destination topicTextDestination;

 

/**

 * 发送文本消息

 * @param text

 */

public void sendTextMessage(final String text){

jmsTemplate.send(topicTextDestination, new MessageCreator() {

public Message createMessage(Session session) throws JMSException {

return session.createTextMessage(text);

}

});

}

}

3)编写测试类

import org.junit.Test;

import org.junit.runner.RunWith;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.test.context.ContextConfiguration;

import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

import cn.itcast.demo.TopicProducer;

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(locations="classpath:applicationContext-activemq-producer.xml")

public class TestTopic {

@Autowired

private TopicProducer topicProducer;

@Test

public void sendTextQueue(){

topicProducer.sendTextMessage();

}

}

3.2.2消息消费者

(1)activemq-spring-consumer工程中建立配置文件applicationContext-jms-consumer-topic.xml

    <!-- 真正能够产生Connection的ConnectionFactory,由对应的 JMS服务厂商提供-->  

<bean id="targetConnectionFactory" class="org.apache.activemq.ActiveMQConnectionFactory">  

    <property name="brokerURL" value="tcp://192.168.25.135:61616"/>  

</bean>

    <!-- Spring用于管理真正的ConnectionFactory的ConnectionFactory -->  

<bean id="connectionFactory" class="org.springframework.jms.connection.SingleConnectionFactory">  

<!-- 目标ConnectionFactory对应真实的能够产生JMS Connection的ConnectionFactory -->  

    <property name="targetConnectionFactory" ref="targetConnectionFactory"/>  

</bean>  

    <!--这个是队列目的地,点对点的  文本信息-->  

<bean id="topicTextDestination" class="org.apache.activemq.command.ActiveMQTopic">  

    <constructor-arg value="topic_text"/>  

</bean>    

<!-- 个人监听类 -->

<bean id="myMessageListener" class="cn.itcast.demo.MyMessageListener"></bean>

<!-- 消息监听容器 -->

<bean class="org.springframework.jms.listener.DefaultMessageListenerContainer">

<property name="connectionFactory" ref="connectionFactory" />

<property name="destination" ref="topicTextDestination" />

<property name="messageListener" ref="myMessageListener" />

</bean>

(2)编写测试类

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(locations="classpath:applicationContext-jms-consumer-topic.xml")

public class TestTopic {

@Test

public void testTopic(){

try {

System.in.read();

} catch (IOException e) {

e.printStackTrace();

}

}

}

测试:同时运行三个消费者工程,在运行生产者工程,查看三个消费者工程的控制台输出。

相关文章
相关标签/搜索