zoukankan      html  css  js  c++  java
  • activeMQ消息队列的使用

    ActiveMQ解决问题:  1.解决服务之间的耦合 2.增加系统并发处理量.

    它使用的是标准生产者和消费者模型.有两种数据结构:Queue/Topic

    1.Queue队列,生产者生产一个消息,只能由一个消费者进行消费.

    2.Topic 话题.生产者生产一个消息,可以由多个消费者进行消费.,

    结合Spring完成ActiveMQ编程

    ActiveMQ 应用场景分析
    1、 用户注册,重点用户信息数据库保存,发短信、发邮件,增加业务处理复杂度,这
    时候使用 MQ, 将发短信、发邮箱,通知 MQ,由另外服务平台完成
    2、 搜索平台、缓存平台
    查询数据,建立缓存、索引 ,不从数据库查询,从缓存或者索引库查询
    当增加、修改、删除数据时,发送消息给 MQ, 缓存平台、索引平台 从 MQ 获取
    到这个信息,更新缓存或者索引

    (一) 导入jar包 

    <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
      <modelVersion>4.0.0</modelVersion>
      <groupId>cn.itcast.maven</groupId>
      <artifactId>activeMQ_spring</artifactId>
      <version>0.0.1-SNAPSHOT</version>
      <name>activeMQ_spring</name>
      
      <dependencies>
    <!-- Spring开发测试--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>4.1.7.RELEASE</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>4.1.7.RELEASE</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> </dependency>
    <!--ActiveMQ 包--> <dependency> <groupId>org.apache.activemq</groupId> <artifactId>activemq-all</artifactId> <version>5.14.0</version> </dependency>
    <!-- Spring整合MQ包--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-jms</artifactId> <version>4.1.7.RELEASE</version> </dependency> </dependencies> </project>

    二  1.编写配置生产者

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
        xmlns:context="http://www.springframework.org/schema/context"
        xmlns:jdbc="http://www.springframework.org/schema/jdbc" xmlns:tx="http://www.springframework.org/schema/tx"
        xmlns:jpa="http://www.springframework.org/schema/data/jpa" xmlns:task="http://www.springframework.org/schema/task"
        xmlns:amq="http://activemq.apache.org/schema/core"
        xmlns:jms="http://www.springframework.org/schema/jms"
        xsi:schemaLocation="
            http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.1.xsd
            http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.1.xsd
            http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.1.xsd
            http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc-4.1.xsd
            http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.1.xsd
            http://www.springframework.org/schema/data/jpa 
            http://www.springframework.org/schema/data/jpa/spring-jpa.xsd
            http://www.springframework.org/schema/jms
            http://www.springframework.org/schema/jms/spring-jms.xsd
            http://activemq.apache.org/schema/core
            http://activemq.apache.org/schema/core/activemq-core-5.8.0.xsd ">
        
        <!-- 扫描包 -->
        <context:component-scan base-package="cn.itcast.activemq" />
        
        <!-- ActiveMQ 连接工厂 -->
        <!-- 真正可以产生Connection的ConnectionFactory,由对应的 JMS服务厂商提供-->
        <!-- 如果连接网络:tcp://ip:61616;未连接网络:tcp://localhost:61616 以及用户名,密码-->
        <amq:connectionFactory id="amqConnectionFactory"
            brokerURL="tcp://localhost:61616" userName="admin" password="admin"  />
    
        <!-- Spring Caching连接工厂 -->
        <!-- Spring用于管理真正的ConnectionFactory的ConnectionFactory -->  
        <bean id="connectionFactory" class="org.springframework.jms.connection.CachingConnectionFactory">
            <!-- 目标ConnectionFactory对应真实的可以产生JMS Connection的ConnectionFactory -->  
            <property name="targetConnectionFactory" ref="amqConnectionFactory"></property>
            <!-- 同上,同理 -->
            <!-- <constructor-arg ref="amqConnectionFactory" /> -->
            <!-- Session缓存数量 -->
            <property name="sessionCacheSize" value="100" />
        </bean>
        
         <!-- Spring JmsTemplate 的消息生产者 start-->
    
        <!-- 定义JmsTemplate的Queue类型 -->
        <bean id="jmsQueueTemplate" class="org.springframework.jms.core.JmsTemplate">
            <!-- 这个connectionFactory对应的是我们定义的Spring提供的那个ConnectionFactory对象 -->  
            <constructor-arg ref="connectionFactory" />
            <!-- 非pub/sub模型(发布/订阅),即队列模式 -->
            <property name="pubSubDomain" value="false" />
        </bean>
    
        <!-- 定义JmsTemplate的Topic类型 -->
        <bean id="jmsTopicTemplate" class="org.springframework.jms.core.JmsTemplate">
             <!-- 这个connectionFactory对应的是我们定义的Spring提供的那个ConnectionFactory对象 -->  
            <constructor-arg ref="connectionFactory" />
            <!-- pub/sub模型(发布/订阅) -->
            <property name="pubSubDomain" value="true" />
        </bean>
    
        <!--Spring JmsTemplate 的消息生产者 end-->
        
        
    </beans>
    applicationContext-mq.xml

    2.完成代码(Queue.Topic)

    @Service
    public class QueueSender {
        // 注入jmsTemplate
        @Autowired
        @Qualifier("jmsQueueTemplate")
        private JmsTemplate jmsTemplate;
        // 匿名内部类中用到外面的属性其,必须被final修饰
        public void send(String queueName, final String message) {
            jmsTemplate.send(queueName, new MessageCreator() {
                public Message createMessage(Session session) throws JMSException {
                    return session.createTextMessage(message);
                }
            });
        }
    }
    @Service
    public class TopicSender {
        // 注入jmsTemplate
        @Autowired
        @Qualifier("jmsTopicTemplate")
        private JmsTemplate jmsTemplate;
    
        public void send(String topicName, final String message) {
            jmsTemplate.send(topicName, new MessageCreator() {
    
                public Message createMessage(Session session) throws JMSException {
                    return session.createTextMessage(message);
                }
            });
        }
    }

    3.测试类

    @RunWith(SpringJUnit4ClassRunner.class)
    @ContextConfiguration(locations = "classpath:applicationContext-mq.xml")
    public class ProducerTest {
        @Autowired
        private QueueSender queueSender;
    
        @Autowired
        private TopicSender topicSender;
    
        @Test
        public void testSendMessage() {
            queueSender.send("spring_queue", "再见,上海");
            topicSender.send("spring_topic", "你好,西安");
        }
    }

    三 编写消费者代码

    1.配置及导包

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
        xmlns:context="http://www.springframework.org/schema/context"
        xmlns:jdbc="http://www.springframework.org/schema/jdbc" xmlns:tx="http://www.springframework.org/schema/tx"
        xmlns:jpa="http://www.springframework.org/schema/data/jpa" xmlns:task="http://www.springframework.org/schema/task"
        xmlns:amq="http://activemq.apache.org/schema/core"
        xmlns:jms="http://www.springframework.org/schema/jms"
        xsi:schemaLocation="
            http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.1.xsd
            http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.1.xsd
            http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.1.xsd
            http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc-4.1.xsd
            http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.1.xsd
            http://www.springframework.org/schema/data/jpa 
            http://www.springframework.org/schema/data/jpa/spring-jpa.xsd
            http://www.springframework.org/schema/jms
            http://www.springframework.org/schema/jms/spring-jms.xsd
            http://activemq.apache.org/schema/core
            http://activemq.apache.org/schema/core/activemq-core-5.8.0.xsd ">
        
        <!-- 扫描包 -->
        <context:component-scan base-package="cn.itcast.activemq.consumer" />
        
        <!-- ActiveMQ 连接工厂 -->
        <!-- 真正可以产生Connection的ConnectionFactory,由对应的 JMS服务厂商提供-->
        <!-- 如果连接网络:tcp://ip:61616;未连接网络:tcp://localhost:61616 以及用户名,密码-->
        <amq:connectionFactory id="amqConnectionFactory"
            brokerURL="tcp://localhost:61616" userName="admin" password="admin"  />
    
        <!-- Spring Caching连接工厂 -->
        <!-- Spring用于管理真正的ConnectionFactory的ConnectionFactory -->  
        <bean id="connectionFactory" class="org.springframework.jms.connection.CachingConnectionFactory">
            <!-- 目标ConnectionFactory对应真实的可以产生JMS Connection的ConnectionFactory -->  
            <property name="targetConnectionFactory" ref="amqConnectionFactory"></property>
            <!-- 同上,同理 -->
            <!-- <constructor-arg ref="amqConnectionFactory" /> -->
            <!-- Session缓存数量 -->
            <property name="sessionCacheSize" value="100" />
        </bean>
        
         <!-- 消息消费者 start-->
    
        <!-- 定义Queue监听器 -->
        <jms:listener-container destination-type="queue" container-type="default" 
            connection-factory="connectionFactory" acknowledge="auto">
            <!-- 默认注册bean名称,应该是类名首字母小写  -->
            <jms:listener destination="spring_queue" ref="queueConsumer1"/>
            <jms:listener destination="spring_queue" ref="queueConsumer2"/>
        </jms:listener-container>
        
        <!-- 定义Topic监听器 -->
        <jms:listener-container destination-type="topic" container-type="default" 
            connection-factory="connectionFactory" acknowledge="auto">
            <jms:listener destination="spring_topic" ref="topicConsumer1"/>
            <jms:listener destination="spring_topic" ref="topicConsumer2"/>
        </jms:listener-container>
    
        <!-- 消息消费者 end -->
        
        
    </beans>

    2.完成代码

    @Service
    public class TopicConsumer1 implements MessageListener {
    
        public void onMessage(Message message) {
            TextMessage textMessage = (TextMessage) message;
            try {
                System.out.println("消费者TopicConsumer1获取消息:" + textMessage.getText());
            } catch (JMSException e) {
                e.printStackTrace();
            }
        }
    
    }
    @Service
    public class QueueConsumer1 implements MessageListener {
        public void onMessage(Message message) {
            // 将message转换成text类型,获得text
            TextMessage textMessage = (TextMessage) message;
            try {
                System.out.println("消费者QueueConsumer1获取消息:" + textMessage.getText());
            } catch (JMSException e) {
                e.printStackTrace();
            }
        }
    }

    2.测试类

    @RunWith(SpringJUnit4ClassRunner.class)
    @ContextConfiguration(locations = "classpath:applicationContext-mq-consumer.xml")
    public class ConsumerTest {
        @Test
        public void testConsumerMessage() {
            while (true) {
                // junit退出,防止进程死掉
            }
        }
    }
  • 相关阅读:
    UVA
    UVA
    UVA
    UVA
    UVA
    UVA
    UVA
    UVA
    UVA
    使用Jmeter(三十)针对ActiveMQ JMS POINT TO POINT压力测试(转载)
  • 原文地址:https://www.cnblogs.com/wwwzzz/p/8081980.html
Copyright © 2011-2022 走看看