赞
踩
MQ的生产者必备的特性有:消息的同步发送,异步发送,消息的ACK与重试机制,消息的顺序生产,批量发送等。RocketMQ在具备这些特性的同时,有自己独有的特性。下面我们对RocketMQ的生产者开展讲解。
1.同步发送消息
同步发送是指消息发送方发出数据后,同步等待,直到收到接收方发回响应之后才发下一个请求。
public class SyncProducer { public static void main(String[] args) throws Exception{ // 实例化消息生产者Producer DefaultMQProducer producer = new DefaultMQProducer("please_rename_unique_group_name"); // 设置NameServer的地址 producer.setNamesrvAddr("127.0.0.1:9876");//106.55.246.66 // 启动Producer实例 producer.start(); for (int i = 0; i < 100; i++) { // 创建消息,并指定Topic,Tag和消息体 Message msg = new Message("TopicTest" /* Topic */, "TagA" /* Tag */, ("Hello RocketMQ " + i).getBytes(RemotingHelper.DEFAULT_CHARSET) /* Message body */ ); // 发送消息到一个Broker SendResult sendResult = producer.send(msg); // 通过sendResult返回消息是否成功送达 System.out.printf("%s%n", sendResult); } // 如果不再发送消息,关闭Producer实例。 producer.shutdown(); } }
2.异步发送消息
消息发送方在发送了一条消息后,不等接收方发回响应,接着进行第二条消息发送。发送方通过回调接口的方式接收服务器响应,并对响应结果进 行处理
public class AsyncProducer { public static void main(String[] args) throws Exception{ // 实例化消息生产者Producer DefaultMQProducer producer = new DefaultMQProducer("please_rename_unique_group_name"); // 设置NameServer的地址 producer.setNamesrvAddr("127.0.0.1:9876");//106.55.246.66 // 启动Producer实例 producer.start(); producer.setRetryTimesWhenSendAsyncFailed(0); //启用Broker故障延迟机制 producer.setSendLatencyFaultEnable(true); for (int i = 0; i < 100; i++) { final int index = i; // 创建消息,并指定Topic,Tag和消息体 Message msg = new Message("TopicTest", "TagA", "OrderID888", "Hello world".getBytes(RemotingHelper.DEFAULT_CHARSET)); // SendCallback接收异步返回结果的回调 producer.send(msg, new SendCallback() { @Override public void onSuccess(SendResult sendResult) { System.out.printf("%-10d OK %s %n", index, sendResult.getMsgId()); } @Override public void onException(Throwable e) { System.out.printf("%-10d Exception %s %n", index, e);e.printStackTrace(); } }); } Thread.sleep(10000); // 如果不再发送消息,关闭Producer实例。 producer.shutdown(); } }
3.单向发送
单向(Oneway)发送特点为发送方只负责发送消息,不等待服务器回应且没有回调函数触发,即只发送请求不等待应答。此方式发送消息的过程耗 时非常短,一般在微秒级别。
public class OnewayProducer { public static void main(String[] args) throws Exception{ // 实例化消息生产者Producer 对象。 DefaultMQProducer producer = new DefaultMQProducer("please_rename_unique_group_name"); // 设置NameServer的地址 producer.setNamesrvAddr("127.0.0.1:9876");//106.55.246.66 // 启动Producer实例 producer.start(); for (int i = 0; i < 20; i++) { // 创建消息,并指定Topic,Tag和消息体 Message msg = new Message("TopicTest" /* Topic */, "TagA" /* Tag */, ("Hello RocketMQ " + i).getBytes(RemotingHelper.DEFAULT_CHARSET) /* Message body */ ); // 发送单向消息,没有任何返回结果 producer.sendOneway(msg); } // 如果不再发送消息,关闭Producer实例。 producer.shutdown(); } }
RocketMQ的顺序性和Kafka一样,主题内每个Queue上的消息是被顺序消费的,而Queue之间的顺序无法保证顺序性。所以需要保证有顺序的消息发送到同一个Queue上。Kafka是通过key值判断消息在哪个分区上,而RocketMQ是生产者直接指定Queue来发送到指定的Queue上。这也避免了Kafka分区再均衡出现同一个key分配到了不同分区上的问题。
public class ProducerInOrder { public static void main(String[] args) throws Exception { DefaultMQProducer producer = new DefaultMQProducer("OrderProducer"); producer.setNamesrvAddr("127.0.0.1:9876");//106.55.246.66 producer.start(); String[] tags = new String[]{"TagA", "TagC", "TagD"}; // 订单列表 List<Order> orderList = new ProducerInOrder().buildOrders(); Date date = new Date(); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); String dateStr = sdf.format(date); for (int i = 0; i < orderList.size(); i++) { // 加个时间前缀 String body = dateStr + " Order:" + orderList.get(i); Message msg = new Message("PartOrder", tags[i % tags.length], "KEY" + i, body.getBytes()); SendResult sendResult = producer.send(msg, new MessageQueueSelector() { @Override public MessageQueue select(List<MessageQueue> mqs, Message msg, Object arg) { Long id = (Long) arg; //根据订单id选择发送queue long index = id % mqs.size(); return mqs.get((int) index); } }, orderList.get(i).getOrderId());//订单id System.out.println(String.format("SendResult status:%s, queueId:%d, body:%s", sendResult.getSendStatus(), sendResult.getMessageQueue().getQueueId(), body)); } producer.shutdown(); } /** * 订单 */ private static class Order { private long orderId; private String desc; public long getOrderId() { return orderId; } public void setOrderId(long orderId) { this.orderId = orderId; } public String getDesc() { return desc; } public void setDesc(String desc) { this.desc = desc; } @Override public String toString() { return "Order{" + "orderId=" + orderId + ", desc='" + desc + '\'' + '}'; } } /** * 生成模拟订单数据 3个订单 每个订单4个状态 */ private List<Order> buildOrders() { List<Order> orderList = new ArrayList<Order>(); Order orderDemo = new Order(); orderDemo.setOrderId(20210406001L); orderDemo.setDesc("创建"); orderList.add(orderDemo); orderDemo = new Order(); orderDemo.setOrderId(20210406002L); orderDemo.setDesc("创建"); orderList.add(orderDemo); orderDemo = new Order(); orderDemo.setOrderId(20210406001L); orderDemo.setDesc("付款"); orderList.add(orderDemo); orderDemo = new Order(); orderDemo.setOrderId(20210406003L); orderDemo.setDesc("创建"); orderList.add(orderDemo); orderDemo = new Order(); orderDemo.setOrderId(20210406002L); orderDemo.setDesc("付款"); orderList.add(orderDemo); orderDemo = new Order(); orderDemo.setOrderId(20210406003L); orderDemo.setDesc("付款"); orderList.add(orderDemo); orderDemo = new Order(); orderDemo.setOrderId(20210406002L); orderDemo.setDesc("推送"); orderList.add(orderDemo); orderDemo = new Order(); orderDemo.setOrderId(20210406003L); orderDemo.setDesc("推送"); orderList.add(orderDemo); orderDemo = new Order(); orderDemo.setOrderId(20210406002L); orderDemo.setDesc("完成"); orderList.add(orderDemo); orderDemo = new Order(); orderDemo.setOrderId(20210406001L); orderDemo.setDesc("推送"); orderList.add(orderDemo); orderDemo = new Order(); orderDemo.setOrderId(20210406003L); orderDemo.setDesc("完成"); orderList.add(orderDemo); orderDemo = new Order(); orderDemo.setOrderId(20210406001L); orderDemo.setDesc("完成"); orderList.add(orderDemo); return orderList; } }
producerGroup:生产者所属组
defaultTopicQueueNums:默认主题在每一个 Broker 队列数量
sendMsgTimeout:发送消息默认超时时间,默认 3s
compressMsgBodyOverHowmuch:消息体超过该值则启用压缩,默认 4k
retryTimesWhenSendFailed:同步方式发送消息重试次数,默认为 2,总共执行 3 次
retryTimesWhenSendAsyncFailed:异步方法发送消息重试次数,默认为 2
retryAnotherBrokerWhenNotStoreOK:消息重试时选择另外一个 Broker 时,是否不等待存储结果就返回,默认为 false
maxMessageSize:允许发送的最大消息长度,默认为 4M
RocketMQ提供了延时消息生产机制。在EMQX中,也有延时消息的机制。可见延时机制在MQ中也占有一定的应用市场。
概念:Producer 将消息发送到消息队列 RocketMQ 服务端,但并不期望这条消息立马投递,而是延迟一定时间后才投递到 Consumer 进行消费, 该消息即延时消息。
Apache RocketMQ 目前只支持固定精度的定时消息,因为如果要支持任意的时间精度,在 Broker 层面,必须要做消息排序,如果再涉及到持久化, 那么消息排序要不可避免的产生巨大性能开销。(阿里云 RocketMQ 提供了任意时刻的定时消息功能,Apache 的 RocketMQ 并没有,阿里并没有开源) 发送延时消息时需要设定一个延时时间长度,消息将从当前发送时间点开始延迟固定时间之后才开始投递。
延迟消息是根据延迟队列的 level 来的,延迟队列默认是msg.setDelayTimeLevel(3)代表延迟 10 秒 “1s 5s 10s 30s 1m 2m 3m 4m 5m 6m 7m 8m 9m 10m 20m 30m 1h 2h”
public class ScheduledMessageProducer { public static void main(String[] args) throws Exception { // 实例化一个生产者来产生延时消息 DefaultMQProducer producer = new DefaultMQProducer("ScheduledProducer"); // 设置NameServer的地址 producer.setNamesrvAddr("39.100.116.73:9876"); // 启动Producer实例 producer.start(); int totalMessagesToSend = 10; for (int i = 0; i < totalMessagesToSend; i++) { Message message = new Message("ScheduledTopic", ("Hello scheduled message " + i).getBytes()); // 设置延时等级3,这个消息将在10s之后投递给消费者(详看delayTimeLevel) // delayTimeLevel:(1~18个等级)"1s 5s 10s 30s 1m 2m 3m 4m 5m 6m 7m 8m 9m 10m 20m 30m 1h 2h" message.setDelayTimeLevel(4); // 发送消息 producer.send(message); } // 关闭生产者 producer.shutdown(); } }
消费者无需特殊处理,普通订阅这个主题即可。
批量发送消息能显著提高传递小消息的性能。限制是这些批量消息应该有相同的 topic,相同的 waitStoreMsgOK,而且不能是延 时消息。此外,这一批消息的总大小不应超过 4MB。
public class BatchProducer { public static void main(String[] args) throws Exception { // 实例化消息生产者Producer DefaultMQProducer producer = new DefaultMQProducer("BatchProducer"); // 设置NameServer的地址 producer.setNamesrvAddr("127.0.0.1:9876"); // 启动Producer实例 producer.start(); String topic = "BatchTest"; List<Message> messages = new ArrayList<>(); messages.add(new Message(topic, "Tag", "OrderID001", "Hello world 0".getBytes())); messages.add(new Message(topic, "Tag", "OrderID002", "Hello world 1".getBytes())); messages.add(new Message(topic, "Tag", "OrderID003", "Hello world 2".getBytes())); try { producer.send(messages); } catch (Exception e) { producer.shutdown(); e.printStackTrace(); } // 如果不再发送消息,关闭Producer实例。 producer.shutdown(); } }
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。