消费者角色:

1. 推式(一般建议用推式)

2. 拉式

消费模式:

1. 集群(cluster)                --均衡负载消费

2. 广播(broadcasting) --发布和订阅者模式

MQ的消费不会清除broker中的数据,broker数据一直存在队列中,队列offset会一直递增,因此可以通过回查来获取到丢失数据。这个时候我们可以采用pull形式较好。

push形式,MQ会记录访问的偏移量,即使宕机下次重启也会按照顺序继续消费,不会出现重复消费。

RocketMQ入门(生产者)_2中已经写过一个推式的代码,接下来就看下拉式。

/**
* 普通拉式消费者,代码编写
* @author DennyZhao
*
*/
public class PullConsumer { /**
* 暂时以map作为offset入库看待。<queueId, offset>
*/
private static Map<String, Long> offsetMap = new HashMap<String, Long>(); public static void main(String[] args) throws UnsupportedEncodingException {
//创建拉式消费者
DefaultMQPullConsumer pullConsumer = new DefaultMQPullConsumer("pullConsumerGroup");
pullConsumer.setNamesrvAddr("192.168.68.137:9876;192.168.68.138:9876;");
try {
pullConsumer.start();
Set<MessageQueue> mqSet= pullConsumer.fetchSubscribeMessageQueues("fruit");
while(true) {
//循环队列
for(MessageQueue mq: mqSet) {
// 从队列中获取固定偏移值
PullResult pullResult = pullConsumer.pullBlockIfNotFound(mq, "*", getOffset(mq), 32);
setOffset(mq, pullResult.getNextBeginOffset());
switch(pullResult.getPullStatus()) {
case FOUND:
List<MessageExt> msgFoundList = pullResult.getMsgFoundList();
for(MessageExt msg : msgFoundList) {
String fruit = new String(msg.getBody(), RemotingHelper.DEFAULT_CHARSET);
System.out.println(fruit + " -----fruit");
}
break;
case NO_NEW_MSG:
break;
case NO_MATCHED_MSG:
break;
}
}
Thread.sleep(2000);
}
} catch (MQClientException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (RemotingException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (MQBrokerException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} /**
* set offset
* @param mq
* @param nextBeginOffset
*/
private static void setOffset(MessageQueue mq, long nextBeginOffset) {
String queueId = mq.getBrokerName() + mq.getTopic() + mq.getQueueId();
offsetMap.put(queueId, nextBeginOffset);
} /**
* 获取固定偏移值
* @param mq queueId
* @return int
*/
private static long getOffset(MessageQueue mq) {
String queueId = mq.getBrokerName() + mq.getTopic() + mq.getQueueId();
Long offset = offsetMap.get(queueId);
if(offset == null) {
offset = 0l;
}
System.out.println(offset + "---------------");
return offset;
} }

使用Schedule拉式:

/**
* ScheduleService 進行數據拉取
* @author DennyZhao
*
*/
public class PullScheduleService { public static void main(String[] args) throws MQClientException {
MQPullConsumerScheduleService scheduleService = new MQPullConsumerScheduleService("scheduleConsumers");
scheduleService.setMessageModel(MessageModel.CLUSTERING);
scheduleService.setPullThreadNums(4);
DefaultMQPullConsumer defaultMQPullConsumer = new DefaultMQPullConsumer("pullConsumer");
defaultMQPullConsumer.setNamesrvAddr("192.168.68.137:9876;192.168.68.138:9876;");
scheduleService.setDefaultMQPullConsumer(defaultMQPullConsumer);
scheduleService.registerPullTaskCallback("fruit", new PullTaskCallback() {
/**
* 数据处理
*/
@Override
public void doPullTask(MessageQueue mq, PullTaskContext context) {
MQPullConsumer pullConsumer = context.getPullConsumer();
try {
long offset = pullConsumer.fetchConsumeOffset(mq, false);
PullResult pull = pullConsumer.pull(mq, "*", offset, 32);
switch(pull.getPullStatus()) {
case FOUND:
// 结果输出
List<MessageExt> msgFoundList = pull.getMsgFoundList();
for(MessageExt msg : msgFoundList) {
String fruit = new String(msg.getBody(), RemotingHelper.DEFAULT_CHARSET);
System.out.println("result: " + fruit);
}
break;
case NO_MATCHED_MSG:
break;
default: }
// 获取下一个循环的offset
pullConsumer.updateConsumeOffset(mq, pull.getNextBeginOffset());
// 设置下次访问时间
context.setPullNextDelayTimeMillis(1000);
} catch (MQClientException | RemotingException | MQBrokerException | InterruptedException | UnsupportedEncodingException e) {
e.printStackTrace();
}
}
});
scheduleService.start();
} }

参数说明:

//push主要参数
DefaultMQPushConsumer pushConsumer = new DefaultMQPushConsumer("pushConsumerGroup");
// 从何地开始,默认(CONSUME_FROM_LAST_OFFSET)
pushConsumer.setConsumeFromWhere(ConsumeFromWhere.CONSUME_FROM_LAST_OFFSET);
pushConsumer.setConsumeThreadMin(2); //最小线程数
pushConsumer.setConsumeThreadMax(8); //最大线程数
pushConsumer.setConsumeTimeout(5000); //连接超时
pushConsumer.setMessageModel(MessageModel.CLUSTERING);//消息模式(集群CLUSTERING和广播BROADCASTING,default:cluster)
pushConsumer.setConsumeConcurrentlyMaxSpan(1000);//单队列最大消费数1000
pushConsumer.setConsumeMessageBatchMaxSize(1); //批量消费数1
pushConsumer.setNamesrvAddr("192.168.68.137:9876;192.168.68.138:9876;");//集群IP
pushConsumer.setHeartbeatBrokerInterval(2000); //心跳监测
pushConsumer.setMaxReconsumeTimes(3);//重复消费次数,用于失败后重试
pushConsumer.queryMessage(topic, key, maxNum, begin, end); //获取消息
pushConsumer.fetchSubscribeMessageQueues(topic);//订阅topic
pushConsumer.registerMessageListener(new MessageListenerConcurrently());//及时普通消费型
pushConsumer.registerMessageListener(new MessageListenerOrderly()); //严格顺序消费型;
        // pull常用参数
//消息模式(集群CLUSTERING和广播BROADCASTING,default:cluster)
pullConsumer.setMessageModel(MessageModel.CLUSTERING);
pullConsumer.fetchSubscribeMessageQueues(topic); //订阅主题
pullConsumer.fetchConsumeOffset(mq, false); //获取queue当前offset位置
pullConsumer.pullBlockIfNotFound(mq, subExpression, offset, maxNums);//获取消费内容
pullConsumer.updateConsumeOffset(mq, offset); //更新消费位置
pullConsumer.setConsumerPullTimeoutMillis(5000); //连接超时

RocketMQ入门(消费者)_3的更多相关文章

  1. RocketMQ入门(2)最佳实践

    转自:http://www.changeself.net/archives/rocketmq入门(2)最佳实践.html RocketMQ入门(2)最佳实践 一.服务端安装部署 我是在虚拟机中的Cen ...

  2. 必须先理解的RocketMQ入门手册,才能再次深入解读

    RocketMQ入门手册 RocketMQ是一个分布式.队列模型的开源消息中间件,前身是MetaQ,是阿里研发的一个队列模型的消息中间件,后开源给apache基金会成为了apache的顶级开源项目,具 ...

  3. RocketMQ入门到入土(二)事务消息&顺序消息

    接上一篇:RocketMQ入门到入土(一)新手也能看懂的原理和实战! 一.事务消息的由来 1.案例 引用官方的购物案例: 小明购买一个100元的东西,账户扣款100元的同时需要保证在下游的积分系统给小 ...

  4. RocketMQ入门(3)拉取消息

    转自:http://www.changeself.net/archives/rocketmq入门(3)拉取消息.html RocketMQ入门(3)拉取消息 RocketMQ不止可以直接推送消息,在消 ...

  5. RocketMQ入门(1)

    转自:http://www.changeself.net/archives/rocketmq入门(1).html RocketMQ入门(1) RocketMQ是一款分布式.队列模型的消息中间件,具有以 ...

  6. RocketMQ之消费者启动与消费流程

    vivo 互联网服务器团队 - Li Kui 一.简介 1.1 RocketMQ 简介 RocketMQ是由阿里巴巴开源的分布式消息中间件,支持顺序消息.定时消息.自定义过滤器.负载均衡.pull/p ...

  7. 消息队列之-RocketMQ入门

    简介 RocketMQ是阿里开源的消息中间件,目前已经捐献个Apache基金会,它是由Java语言开发的,具备高吞吐量.高可用性.适合大规模分布式系统应用等特点,经历过双11的洗礼,实力不容小觑. 官 ...

  8. 轻松搞定RocketMQ入门

    RocketMQ是一款分布式.队列模型的消息中间件,具有以下特点: 能够保证严格的消息顺序 提供丰富的消息拉取模式 高效的订阅者水平扩展能力 实时的消息订阅机制 亿级消息堆积能力 RocketMQ网络 ...

  9. RocketMQ入门(生产者)_2

    从 RocketMQ环境搭建_1 我们已经建立了MQ的Server,接下来就是简单的生产和消费的过程. 1. rocketMQ的源码中有个示例代码example  ,我们从Apache官网中可以下载源 ...

随机推荐

  1. Oracle Base64加解密

    参考 http://blog.csdn.net/liuzhigang1237/article/details/7591439

  2. Ubuntu 终端使用ss代理

    用polipo软件,这个软件可以吧socket5转换成http代理 $ sudo apt-get install polipo $ sudo vim /etc/polipo/config 在文件中加入 ...

  3. 2018-2019-2 20165308网络对抗技术 Exp6:信息收集与漏洞扫描

    2018-2019-2 20165308网络对抗技术 Exp6:信息收集与漏洞扫描 实践目标 掌握信息搜集的最基础技能与常用工具的使用方法. 实践内容 (1)各种搜索技巧的应用 (2)DNS IP注册 ...

  4. oracle 的 exp 和imp命令

    数据导出: 1 将数据库TEST完全导出,用户名gdoa 密码123 导出到D:\TEST_BK.dmp中 exp gdoa/123@TEST file=d:\TEST_BK.dmp full=y   ...

  5. iOS Simulator 模拟器 与 Android Emulator 仿真器:为什么叫不同的英文名字?(待补充)

    iOS Simulator 模拟器 与 Android Emulator 仿真器:为什么叫不同的英文名字?(待补充)

  6. update_engine-整体结构(二)

    在update_engine-整体结构(一)中分析UpdateEngineDaemon::OnInit()的整体情况.下面先分析在该方法中涉及的DaemonStateAndroid和BinderUpd ...

  7. 错误界面 SQL2008备份集中的数据库备份与现有的数据库不同,错误号码:3154。

    转载来源  https://blog.csdn.net/zaocha321/article/details/8466735 因为我也用这个方法成功了.所以记录下来 错误界面 SQL2005备份集中的数 ...

  8. 学习笔记《Java多线程编程实战指南》二

    2.1线程属性 属性 属性类型及用途  只读属性  注意事项 编号(id) long型,标识不同线程  是  不适合用作唯一标识 名称(name) String型,区分不同线程  否  设置名称有助于 ...

  9. com.jakewharton:butterknife:7.0.1' 点击无效

    需要加上 annotationProcessor 'com.jakewharton:butterknife:7.0.1' dependencies { compile 'com.jakewharton ...

  10. xshell完美开源替代方案(Kitty+MTPuTTY并设置全局字体)

    xshell是收费的,过了30天就不能用了.我们应该找一个开源的替代品.说实话windows平台没有什么可选的,就是putty.但是原生的putty不好用,记不住密码,又不支持多标签. Kitty是基 ...