mq--activeMQ

概述:

作用:

  1. 解决业务耦合问题,实现异步消息处理
  2. 解决高并发消息处理问题

使用原则:

    不需要立即返回结果的,可以异步处理的。就可以用MQ方式。

    但如果需要立即返回结果,必须同步处理的,就得用Webservice。

 

ActiveMQ的下载安装运行

activemq的不同版本对jdk有要求:

  • 5.15.x以后的版本需要jdk1.8
  • 5.14之前的版本,只需要jdk1.7(1.7的版本也不能太老)

下载地址: http://activemq.apache.org/components/classic/download/

 

安装 :

解压到一个目录

  • bin:可执行文件
  • conf:配置
  • data:activemq数据持久化存储地方kaha
  • docs:文档
  • example:例子
  • lib:软件运行的jar
  • webapps:在线控制台
  • activemq-all.jar:开发jar

运行 :

根据操作系统和jdk的位数选择该命令执行:

第一次启动会初始化。

控制台:

http://localhost:8161

登录的用户名和密码:admin/admin

点击输入 用户名和密码

项目内配置端口地址

 

ActiveMQ 中的生产者和消费者模型两种消息结构

ActiveMQ使用的是标准的生产者和消费者模型。

ActiveMQ支持两种消息结构:

  • Queue:队列消息

  • Topic:话题消息-订阅消息

小结:

  1. 队列消息:一个消息只能由一个消费者消费,一个消费者可以消费多个消息
  2. 话题消息(订阅消息):一个消息可以由多个消费者消费,但前提是必须先订阅类该消息才行。

Spring整合方式实现生产者和消费者

jar包依赖

  <!-- activemq -->
        <dependency>
            <groupId>org.apache.activemq</groupId>
            <artifactId>activemq-all</artifactId>
            <version>5.14.5</version>
        </dependency>
        <!-- spring整合MQ -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-jms</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>

 

直接配置Spring配置文件:配置mq的连接、模版对象

<?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: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/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 ">

    <!-- 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>

生产者

package cn.bufanli.controller;

import cn.bufanli.pojo.DeviceInfo;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.jms.core.JmsTemplate;
import org.springframework.jms.core.MessageCreator;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.Session;
import javax.jms.TextMessage;

/**
 * @author BuShuangLi
 * @date 2019/1/21
 */
@RestController
@RequestMapping("test")
public class TestController {

	//注入模版对象
	//注入queue的消息模版对象
	@Autowired
	@Qualifier("jmsQueueTemplate")//必须根据名字注入
	private JmsTemplate jmsQueueTemplate;
	//注入topic的消息模版对象
	@Autowired
	@Qualifier("jmsTopicTemplate")//必须根据名字注入
	private JmsTemplate jmsTopicTemplate;

	@RequestMapping("producerTest ")
	public void producerTest() {
	//循环发送10条消息
		for (int i = 0; i < 10; i++) {
			final int j = i;
		//发queue的消息
		//参数1:队列的名字,取的时候要用
		//参数2:消息创建者(spring封装对象)
		//匿名内部类写法
		jmsQueueTemplate.send("spring.queue.hello", new MessageCreator() {
			@Override
			public Message createMessage(Session session) throws JMSException {
				TextMessage message = session.createTextMessage("Queue消息:spring整合后的helloworld! 序号: "+j+"  ");
				return message;
			}
		});
		//lambda表达式写法 存储 map
		jmsQueueTemplate.send("Queue.Lambda.hello",s -> {
			MapMessage mapMessage = s.createMapMessage();
			mapMessage.setString("Queue.lambda","测试value 序号: "+j+"  ");
			return mapMessage;
		});
		}
		//发t的消息
		jmsTopicTemplate.send("spring.topic.hello", new MessageCreator() {
			@Override
			public Message createMessage(Session session) throws JMSException {
				TextMessage message = session.createTextMessage("Topic消息:spring整合后的helloworld!");
				return message;
			}
		});

		jmsTopicTemplate.send("Topic.Lambda.hello",s->{
			MapMessage mapMessage = s.createMapMessage();
			//存储map  mapMessage.setString("key","value");
			mapMessage.setString("Topic.Lambda","测试Topiclambda");
			return mapMessage;
		});
	}


}

消费者

第一步:先编写消息处理对象(消费者)

目标:弄两个队列消费者对象和两个话题的消费者对象

四个类内容一样

package cn.bufanli.activemqs.impl.consumer;

import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.MessageListener;
import javax.jms.TextMessage;

/**
 * @author BuShuangLi
 * @date 2019/4/15
 * q的消费者1:处理消息内容
 */
public class QueryConsumer1  implements MessageListener {
	@Override
	public void onMessage(Message message) {
		String str;
		try {
			str = ((TextMessage)message).getText();
			System.err.println(str+"Q1--处理人:"+this.getClass().getSimpleName());
		} catch (JMSException e) {
			e.printStackTrace();
		}

	}
}

Lambda 表达式map 接收方法

package cn.bufanli.activemqs.impl.consumer;

import javafx.beans.property.StringProperty;

import javax.jms.*;
import java.io.Serializable;

/**
 * @author BuShuangLi
 * @date 2019/4/15
 */
public class QueryConsumerLambda2 implements MessageListener {
	@Override
	public void onMessage(Message message) {
		MapMessage mapMessage = (MapMessage) message;
		try {
			Object object = mapMessage.getObject("Queue.lambda");
			System.err.println(object+"Q2--处理人:"+this.getClass().getSimpleName());
		} catch (JMSException e) {
			e.printStackTrace();
		}
	}
}

 

mq-consumer-Context.xml

<?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: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/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 ">

    <!-- 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-->
    <!-- 定义Queue监听器 -->
    <jms:listener-container destination-type="queue" container-type="default" connection-factory="connectionFactory" acknowledge="auto">
        <!-- 注入要处理的消息队列的名字,和要处理的bean对象 -->
        <jms:listener destination="spring.queue.hello" ref="queueConsumer1"/>
        <jms:listener destination="spring.queue.hello" 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.hello" ref="topicConsumer1"/>
        <jms:listener destination="spring.topic.hello" ref="topicConsumer2"/>
    </jms:listener-container>

    <!-- 消息消费者 end -->
    <bean id="queueConsumer1" class="cn.bufanli.activemqs.impl.consumer.QueryConsumer1"/>
    <bean id="queueConsumer2" class="cn.bufanli.activemqs.impl.consumer.QueryConsumer2"/>
    <bean id="topicConsumer1" class="cn.bufanli.activemqs.impl.consumer.TopicConsumer1"/>
    <bean id="topicConsumer2" class="cn.bufanli.activemqs.impl.consumer.TopicConsumer2"/>
</beans>

控制台打印

Queue消息:spring整合后的helloworld! 序号: 0  Q1--处理人:QueryConsumer1
Queue消息:spring整合后的helloworld! 序号: 1  Q1--处理人:QueryConsumer1
Queue消息:spring整合后的helloworld! 序号: 2  Q1--处理人:QueryConsumer1
Queue消息:spring整合后的helloworld! 序号: 3  Q1--处理人:QueryConsumer1
Queue消息:spring整合后的helloworld! 序号: 4  Q1--处理人:QueryConsumer1
Queue消息:spring整合后的helloworld! 序号: 5  Q1--处理人:QueryConsumer1
Queue消息:spring整合后的helloworld! 序号: 6  Q1--处理人:QueryConsumer1
Queue消息:spring整合后的helloworld! 序号: 7  Q1--处理人:QueryConsumer1
Queue消息:spring整合后的helloworld! 序号: 8  Q1--处理人:QueryConsumer1
Queue消息:spring整合后的helloworld! 序号: 9  Q1--处理人:QueryConsumer1

问题:

1)为什么只有Q的消息处理,没有T的消息处理呢?

原因是:T的消息需要先订阅,再消费。订阅之前的消息,不能消费。

而Q的消息无需订阅,直接消费。

2) 为什么只有QueueConsumer2消费了Q的消息?

原因:QueueConsumer2有能力在短时间内把消息都处理完,而不需要其他人。

再做一个边生产边消费效果:

在打开消费监听的情况下,执行生产代码。

发现:

1)Q的消息只能由一个消费者消费,消费过的不能再消费。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: qt activemq-cpp-library-3.9.5-src.zip是一个QT版本的ActiveMQ C++库的压缩包。ActiveMQ是一个流行的开源消息中间件,而C++库是用于在C++应用程序中使用ActiveMQ的工具。 该压缩包包含了使用QT框架开发的ActiveMQ C++库的源代码。使用这个库可以方便地在QT应用程序中发送和接收ActiveMQ消息。它提供了一套易于使用的API,使得开发人员可以快速地集成ActiveMQ功能到他们的应用程序中。 这个库的版本号是3.9.5,这意味着它是在3.9.5版本的ActiveMQ基础上进行开发和定制的。版本号的更新通常会带来新功能、改进和错误修复,因此使用较新版本的库可以提供更好的性能和功能。 要使用这个库,首先需要下载并解压压缩包。然后,可以将源代码导入到QT项目中,并在项目配置中添加必要的依赖项。开发人员可以使用库中提供的API来连接到ActiveMQ代理服务器,发送和接收消息。可以进行一些高级设置,如设置消息过滤器、持久性订阅等。 总结起来,qt activemq-cpp-library-3.9.5-src.zip是一个由QT框架开发的ActiveMQ C++库的源代码压缩包。它提供了便捷的方式在QT应用程序中使用ActiveMQ,并能够发送、接收消息。使用这个库可以提供更好的性能和功能,并且使用较新的版本可以获得更多的更新和改进。 ### 回答2: qt activemq-cpp-library-3.9.5-src.zip 是一个包含 Qt 平台下的 ActiveMQ-CPP 库源代码的压缩文件。ActiveMQ-CPP 是 Apache ActiveMQ 的 C++ 客户端库,为开发人员提供了在 C++ 环境中与 ActiveMQ 通信的能力。 这个库的版本是3.9.5,它是根据 Apache ActiveMQ 的版本定制的。ActiveMQ 是一个开源的消息代理,用于在分布式系统中进行异步通信和消息传递。ActiveMQ-CPP 库提供了在 C++ 应用程序中使用 ActiveMQ 的接口和功能,让开发人员能够使用 C++ 编程语言进行异步消息传递。 该压缩文件包含了 ActiveMQ-CPP 库的源代码,这意味着您可以查看和修改源代码以满足您的特定需求。Qt 是一个跨平台的应用程序开发框架,它可以方便地用于构建图形用户界面和可移植的应用程序。ActiveMQ-CPP 库结合了 ActiveMQ 和 Qt 的功能,为开发人员提供了一种使用 C++ 和 Qt 进行消息传递的快捷方式。 要使用这个库,您需要解压缩该压缩文件,并将源代码文件导入到您的 Qt 项目中。然后,您可以根据活动MQ-CPP 文档中提供的指南来配置和使用该库。 总之,qt activemq-cpp-library-3.9.5-src.zip 是一个包含 ActiveMQ-CPP 库源代码的压缩文件,它让使用 C++ 和 Qt 的开发人员能够在 Qt 平台上与 ActiveMQ 进行异步消息传递。 ### 回答3: qt activemq-cpp-library-3.9.5-src.zip 是一个压缩包,其中包含了 Qt 平台下的 ActiveMQ-CPP 库的源代码。 ActiveMQ-CPP 是一个 C++ 编写的 ActiveMQ 客户端库,用于实现与 ActiveMQ 消息代理服务器的连接和消息交互。它提供了一组用于发送、接收和处理消息的类和函数。使用 ActiveMQ-CPP,开发者可以在 Qt 平台上轻松地实现与 ActiveMQ 服务器的通信,用于实现消息传递的功能。 这个压缩包包含了库的源代码,可以方便地进行自定义和扩展。通过解压这个压缩包,开发者可以获取到库的源代码文件,其中包含了用于实现不同功能的类、函数和头文件。开发者可以根据自己的需要对源代码进行修改和定制,以满足项目的特定需求。 使用这个压缩包,开发者可以通过 Qt 平台来构建和编译 ActiveMQ-CPP 库,生成对应的库文件,然后将其链接到自己的项目中。这样,开发者就可以在自己的 Qt 项目中使用 ActiveMQ-CPP 库提供的功能,实现与 ActiveMQ 服务器的连接和消息交互。 总之,qt activemq-cpp-library-3.9.5-src.zip 是一个包含了 ActiveMQ-CPP 库源代码的压缩包,可以方便地在 Qt 平台上进行定制和扩展,并实现与 ActiveMQ 服务器的通信和消息处理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值