Spring整合RocketMQ
本地部署:
window配置启动:
1、 添加环境变量
ROCKETMQ_HOME="D:\rocketmq"
NAMESRV_ADDR="localhost:9876"
2、启动名称服务器
bin\mqnamesrv.cmd
3、启动Broker
bin\mqbroker.cmd -n localhost:9876 autoCreateTopicEnable=true
可视化控制台:https://github.com/apache/rocketmq-externals/tree/master/rocketmq-console
mvn打包: mvn clean package -Dmaven.test.skip=true
启动:java -jar rocketmq-console-ng-2.0.0.jar
访问:localhost:8080
Spring整合RocketMQ
https://github.com/apache/rocketmq/blob/master/docs/cn/RocketMQ_Example.md
1、依赖
<!--rocketmq--> <dependency> <groupId>org.apache.rocketmq</groupId> <artifactId>rocketmq-client</artifactId> <version>4.8.0</version> </dependency>
2、配置生产者和消费者
@Configuration public class RocketmqConfig { /** * 实例化消息生产者Producer * start()方法用于启动Producer实例 * shutdown()方法关闭Producer实例 */ @Bean(initMethod = "start", destroyMethod = "shutdown") public MQProducer mqProducer() throws MQClientException { DefaultMQProducer producer = new DefaultMQProducer("custom-rocketmq-producer"); // 设置NameServer的地址 producer.setNamesrvAddr("localhost:9876"); // 启动Producer实例 // producer.start(); return producer; } /** * 消费者配置 * * @return * @throws MQClientException */ @Bean public MQPushConsumer mqPushConsumer() throws MQClientException { // 实例化消费者 DefaultMQPushConsumer consumer = new DefaultMQPushConsumer("custom-rocketmq-consumer"); // 设置NameServer的地址 consumer.setNamesrvAddr("localhost:9876"); // 订阅一个或者多个Topic,以及Tag来过滤需要消费的消息 consumer.subscribe("custom-topic1", "*"); // 注册监听器来处理消息,MyConsumer:自定义消息监听器 consumer.registerMessageListener(new MyConsumer()); // 启动消费者实例 consumer.start(); return consumer; } }
3、生产消息示例
@Autowired private MQProducer mqProducer; @Test void producerTest() throws RemotingException, MQClientException, InterruptedException, MQBrokerException { // 同步发送消息 Message syncMsg = new Message("custom-topic1", "sync_msg", "hello RocketMQ syncMsg to custom-topic1".getBytes()); mqProducer.send(syncMsg); // 异步发送消息 Message asyncMsg = new Message("custom-topic1", "async_msg", "hello RocketMQ asyncMsg to custom-topic1".getBytes()); mqProducer.send(asyncMsg, new SendCallback() { @Override public void onSuccess(SendResult sendResult) { // do somethind } @Override public void onException(Throwable e) { // do somethind } }); // 发送延迟消息 Message delayMsg = new Message("custom-topic1", "delay_msg", ("Hello RocketMQ delayMsg to custom-topic1").getBytes()); // 设置延时等级5,这个消息将在1分钟之后收到(现在只支持固定的几个时间,详看delayTimeLevel) delayMsg.setDelayTimeLevel(5); // 发送消息 mqProducer.send(delayMsg); }
4、消费消息
/** * 自定义消息监听器 */ public class MyConsumer implements MessageListenerConcurrently { @Override public ConsumeConcurrentlyStatus consumeMessage(List<MessageExt> messageExts, ConsumeConcurrentlyContext consumeConcurrentlyContext) { for (MessageExt messageExt : messageExts) { System.out.println("Receive New Messages:" + new String(messageExt.getBody())); } return ConsumeConcurrentlyStatus.CONSUME_SUCCESS; } }
SpringBoot整合RocketMQ
https://github.com/apache/rocketmq-spring/wiki
要求:
JDK1.8及以上
Maven 3.0及以上
SpringBoot 2.0及以上
1、添加依赖(rocketmq-client 版本为4.8.0)
<!--add dependency in pom.xml--> <dependency> <groupId>org.apache.rocketmq</groupId> <artifactId>rocketmq-spring-boot-starter</artifactId> <version>2.2.0</version> </dependency>
2、配置namesrv 和 生产组
## application.properties
#支持配置多个nameserver地址,采用;分隔即可
rocketmq.name-server=127.0.0.1:9876
rocketmq.producer.group=my-group
3、发送消息
@Component public class TemplateProducer { @Autowired private RocketMQTemplate rocketMQTemplate; public void producer() { // 同步发送消息 rocketMQTemplate.convertAndSend("test", "Hello, World sync!"); // 异步发送消息 rocketMQTemplate.asyncSend("test", "Hello, World async!", new SendCallback() { @Override public void onSuccess(SendResult sendResult) { System.out.printf("async onSucess SendResult=%s %n", sendResult); } @Override public void onException(Throwable e) { System.out.printf("async onException Throwable=%s %n", e); } }); // 发送消息时指定TAG,通过destination参数指定,格式为topicName:tagName rocketMQTemplate.convertAndSend("test:tag1", "hello,msg of tag1"); //send spring message,Message参数包为: org.springframework.messaging.Message rocketMQTemplate.send("test", MessageBuilder.withPayload("Hello, World! I'm from spring message").build()); // 发送消息时如何设置消息的key,可以通过重载的xxxSend(String destination, Message<?> msg, ...)方法来发送消息,指定msg的headers来完成 String msgId = UUID.randomUUID().toString(); rocketMQTemplate.send("topic-test", MessageBuilder.withPayload("msg of key").setHeader(MessageConst.PROPERTY_KEYS, msgId).build()); // 发送顺序排序消息 rocketMQTemplate.syncSendOrderly("orderly_topic", MessageBuilder.withPayload("Hello, World").build(), "hashkey"); // 同步发送延迟消息,延迟levle为9,即5分钟。 rocketMQTemplate.syncSend("delay-topic", MessageBuilder.withPayload("hello delay msg").build(), 1000, 9); // 销毁rocketMQTemplate,注意:一旦销毁,就不能再使用rocketMQTemplate发送消息 // 不需要手动执行此方法,rocketMQTemplate会在spring容器销毁时自动销毁 //rocketMQTemplate.destroy(); } }
4、消费消息
1)Push模式
/** * push模式消费消息 * RocketMQListener<String> 只获取消息payload * RocketMQListener<MessageExt> 获取rocketmq原生的MessageExt消息 */ @Component @RocketMQMessageListener(topic = "custom-topic1", consumerGroup = "my-consumer_delay-test-1") public class PushConsumer implements RocketMQListener<MessageExt> { @Override public void onMessage(MessageExt messageExt) { System.out.println(("received message: " + new String(messageExt.getBody()))); } }
2)Pull模式
从RocketMQ Spring 2.2.0开始,RocketMQ Spring支持Pull模式消费
①:application.properties 添加配置
rocketmq.consumer.group=my-group1
rocketmq.consumer.topic=test
②:代码中主动拉取消息示例
/** * pull模式消费消息 */ @Component public class PullConsumer { @Resource private RocketMQTemplate rocketMQTemplate; public void pullMessage() { //This is an example of pull consumer using rocketMQTemplate. List<String> messages = rocketMQTemplate.receive(String.class); System.out.printf("receive from rocketMQTemplate, messages=%s %n", messages); } }
项目实践:
1、配置文件:rocketmq.properties
access_key=xxx
secret_key=xxx
name_server_addr=xxx
producer_group=monitor_producer
consumer_group=monitor_consumer
topic_name=high_efficient_movie
2、生产者、消费者配置:RocketmqConfig
/**
* Spring整合rocketmq配置
*
* @author yangyongjie
*/
@Configuration
public class RocketmqConfig implements ApplicationRunner {
private static final Logger LOGGER = LoggerFactory.getLogger(RocketmqConfig.class);
@Resource
private ApplicationContext applicationContext;
/**
* 实例化消息生产者Producer
* start()方法用于启动Producer实例,会被自动调用
* shutdown()方法关闭Producer实例,会被自动调用
*/
@Bean(initMethod = "start", destroyMethod = "shutdown")
public AbstractProducer producer() {
Properties properties = new Properties();
//初始化融合云权限验证(从融合云获取 AK-SK)
properties.setProperty(ConfigKey.ACCESS_KEY, PropertiesUtil.get("access_key"));
properties.setProperty(ConfigKey.SECRET_KEY, PropertiesUtil.get("secret_key"));
//初始化 Producer(指定生产者组名称)
properties.setProperty(ConfigKey.PRODUCER_GROUP, PropertiesUtil.get("producer_group"));
//设置集群的 Namesrv 地址(从 产品简介-集群信息 中查看)
properties.setProperty(ConfigKey.NAME_SERVER_ADDR, PropertiesUtil.get("name_server_addr"));
NormalProducer producer = ClientFactory.createNormalProducer(properties);
LOGGER.info("生产者实例化完成");
return producer;
}
/**
* 初始化消费者
*
* @throws TException
*/
@Override
public void run(ApplicationArguments args) throws Exception {
Map<String, MessageListenerConcurrently> messageListenerMap = applicationContext.getBeansOfType(MessageListenerConcurrently.class);
for (Map.Entry<String, MessageListenerConcurrently> messageListenerEntry : messageListenerMap.entrySet()) {
String topicName = messageListenerEntry.getKey();
MessageListenerConcurrently messageListener = messageListenerEntry.getValue();
// 使用topicName和MessageProcessorFactory消息处理工厂构造消费者
pushConsumer(topicName, messageListener);
}
}
/**
* 消费者配置
*
* @return
* @throws MQClientException
*/
public AbstractConsumer pushConsumer(String topicName, MessageListenerConcurrently messageListener) throws MQClientException {
Properties properties = new Properties();
//初始化融合云权限验证(从融合云获取 AK-SK)
properties.setProperty(ConfigKey.ACCESS_KEY, PropertiesUtil.get("access_key"));
properties.setProperty(ConfigKey.SECRET_KEY, PropertiesUtil.get("secret_key"));
//初始化 Consumer(指定消费者组名称)
properties.setProperty(ConfigKey.CONSUMER_GROUP, PropertiesUtil.get("consumer_group"));
//设置集群的 Namesrv 地址(从 产品简介-集群信息 中查看)
properties.setProperty(ConfigKey.NAME_SERVER_ADDR, PropertiesUtil.get("name_server_addr"));
//创建consumer并注册回调,以便在从代理获取的消息到达时执行。
NormalConsumer consumer = ClientFactory.createNormalConsumer(properties, messageListener);
//订阅topic
consumer.subscribe(topicName, "*");
// 启动消费者实例
consumer.start();
return consumer;
}
}
消费者:
/**
* 高时效节目监听
*
* @author yangyongjie
*/
@Component("high_efficient_movie")
public class HignEfficientMovieConsumer implements MessageListenerConcurrently {
@Override
public ConsumeConcurrentlyStatus consumeMessage(List<MessageExt> list, ConsumeConcurrentlyContext consumeConcurrentlyContext) {
return null;
}
}
END.