从本节开始称Sender为生产者 , Recv为消费者

 

一、消息确认

为了确保消息一定被消费者处理,rabbitMQ提供了消息确认功能,就是在消费者处理完任务之后,就给服务器一个回馈,服务器就会将该消息删除,如果消费者超时不回馈,那么服务器将就将该消息重新发送给其他消费者

默认是开启的,在消费者端通过下面的方式开启消息确认,  首先将autoAck自动确认关闭,等我们的任务执行完成之后,手动的去确认,类似JDBC的autocommit一样

  1.  
  1. QueueingConsumer consumer = new QueueingConsumer(channel);
  2. boolean autoAck = false;
  3. channel.basicConsume("hello", autoAck, consumer);
  1.  

在前面的例子中使用的是channel.basicConsume(channelName, true, consumer) ; 在接收到消息后,就会自动反馈一个消息给服务器。

下面这个例子来测试消息确认的功能。

Sender03.java

  1. package com.zf.rabbitmq03;
  2.  
  3. import java.io.IOException;
  4.  
  5. import com.rabbitmq.client.Channel;
  6. import com.rabbitmq.client.Connection;
  7. import com.rabbitmq.client.ConnectionFactory;
  8.  
  9. /**
  10. * 发送消息
  11. * @author zhoufeng
  12. *
  13. */
  14. public class Sender03 {
  15.  
  16. public static void main(String[] args) throws IOException {
  17.  
  18. ConnectionFactory connFac = new ConnectionFactory() ;
  19.  
  20. //RabbitMQ-Server安装在本机,所以直接用127.0.0.1
  21. connFac.setHost("127.0.0.1");
  22.  
  23. //创建一个连接
  24. Connection conn = connFac.newConnection() ;
  25.  
  26. //创建一个渠道
  27. Channel channel = conn.createChannel() ;
  28.  
  29. //定义Queue名称
  30. String queueName = "queue01" ;
  31.  
  32. //为channel定义queue的属性,queueName为Queue名称
  33. channel.queueDeclare( queueName , false, false, false, null) ;
  34.  
  35. String msg = "Hello World!";
  36.  
  37. //发送消息
  38. channel.basicPublish("", queueName , null , msg.getBytes());
  39.  
  40. System.out.println("send message[" + msg + "] to "+ queueName +" success!");
  41.  
  42. channel.close();
  43. conn.close();
  44.  
  45. }
  46.  
  47. }

与Sender01.java一样,没有什么区别。

Recv03.java

  1. package com.zf.rabbitmq03;
  2.  
  3. import java.io.IOException;
  4.  
  5. import com.rabbitmq.client.Channel;
  6. import com.rabbitmq.client.Connection;
  7. import com.rabbitmq.client.ConnectionFactory;
  8. import com.rabbitmq.client.ConsumerCancelledException;
  9. import com.rabbitmq.client.QueueingConsumer;
  10. import com.rabbitmq.client.QueueingConsumer.Delivery;
  11. import com.rabbitmq.client.ShutdownSignalException;
  12.  
  13. /**
  14. * 接收消息
  15. * @author zhoufeng
  16. *
  17. */
  18. public class Recv03 {
  19.  
  20. public static void main(String[] args) throws IOException, ShutdownSignalException, ConsumerCancelledException, InterruptedException {
  21.  
  22. ConnectionFactory connFac = new ConnectionFactory() ;
  23.  
  24. connFac.setHost("127.0.0.1");
  25.  
  26. Connection conn = connFac.newConnection() ;
  27.  
  28. Channel channel = conn.createChannel() ;
  29.  
  30. String channelName = "channel01";
  31.  
  32. channel.queueDeclare(channelName, false, false, false, null) ;
  33.  
  34. //配置好获取消息的方式
  35. QueueingConsumer consumer = new QueueingConsumer(channel) ;
  36.  
  37. //取消 autoAck
  38. boolean autoAck = false ;
  39.  
  40. channel.basicConsume(channelName, autoAck, consumer) ;
  41.  
  42. //循环获取消息
  43. while(true){
  44.  
  45. //获取消息,如果没有消息,这一步将会一直阻塞
  46. Delivery delivery = consumer.nextDelivery() ;
  47.  
  48. String msg = new String(delivery.getBody()) ;
  49.  
  50. //确认消息,已经收到
  51. channel.basicAck(delivery.getEnvelope().getDeliveryTag()
  52. , false);
  53.  
  54. System.out.println("received message[" + msg + "] from " + channelName);
  55. }
  56.  
  57. }
  58.  
  59. }

注意:一旦将autoAck关闭之后,一定要记得处理完消息之后,向服务器确认消息。否则服务器将会一直转发该消息

如果将上面的channel.basicAck(delivery.getEnvelope().getDeliveryTag(), false);注释掉, Sender03.java只需要运行一次 , Recv03.java每次运行将都会收到HelloWorld消息

注意:

但是这样还是不够的,如果rabbitMQ-Server突然挂掉了,那么还没有被读取的消息还是会丢失 ,所以我们可以让消息持久化。 只需要在定义Queue时,设置持久化消息就可以了,方法如下:

  1.  
  1. boolean durable = true;
  2. channel.queueDeclare(channelName, durable, false, false, null);
  1.  

这样设置之后,服务器收到消息后就会立刻将消息写入到硬盘,就可以防止突然服务器挂掉,而引起的数据丢失了。  但是服务器如果刚收到消息,还没来得及写入到硬盘,就挂掉了,这样还是无法避免消息的丢失。

 

二、公平调度

上一个例子能够实现发送一个Message与接收一个Message

从上一个Recv01中可以看出,必须处理完一个消息,才会去接收下一个消息。如果生产者众多,那么一个消费者肯定是忙不过来的。此时就可以用多个消费者来对同一个Channel的消息进行处理,并且要公平的分配任务给多个消费者。不能部分很忙  部分总是空闲

实现公平调度的方式就是让每个消费者在同一时刻会分配一个任务。 通过channel.basicQos(1);可以设置

rabbitMQ学习笔记(三) 消息确认与公平调度消费者的更多相关文章

  1. RabbitMQ 消息确认与公平调度消费者

    一.消息确认 为了确保消息一定被消费者处理,rabbitMQ提供了消息确认功能,就是在消费者处理完任务之后,就给服务器一个回馈,服务器就会将该消息删除,如果消费者超时不回馈,那么服务器将就将该消息重新 ...

  2. rabbitMQ学习笔记(五) 消息路由

    生产者会生产出很多消息 , 但是不同的消费者可能会有不同的需求,只需要接收指定的消息,其他的消息需要被过滤掉. 这时候就可以对消息进行过滤了. 在消费者端设置好需要接收的消息类型. 如果不使用默认的E ...

  3. Linux内核学习笔记(7)--完全公平调度(CFS)

    一.完全公平调度算法 完全公平调度 CFS 的出发点基于一个简单的理念:进程调度的效果应该如同系统具备一个理想中的完美多任务处理器.在这种系统中,每个进程能够获得 1/n 的处理器时间(n 为可运行进 ...

  4. RabbitMQ学习笔记三:Java实现RabbitMQ之与Spring集成

    搭建好maven项目环境,加入RabbitMQ依赖包 <dependency> <groupId>org.springframework.amqp</groupId> ...

  5. RabbitMQ学习笔记五:RabbitMQ之优先级消息队列

    RabbitMQ优先级队列注意点: 1.只有当消费者不足,不能及时进行消费的情况下,优先级队列才会生效 2.RabbitMQ3.5以后才支持优先级队列 代码在博客:RabbitMQ学习笔记三:Java ...

  6. RabbitMQ学习系列三-C#代码接收处理消息

    RabbitMQ学习系列三:.net 环境下 C#代码订阅 RabbitMQ 消息并处理 http://www.80iter.com/blog/1438251320680361 http://www. ...

  7. rabbitmq学习(三) —— 工作队列

    工作队列,又称任务队列,主要思想是避免立即执行资源密集型任务,并且必须等待完成.相反地,我们进行任务调度,我们将一个任务封装成一个消息,并将其发送到队列.工作进行在后台运行不断的从队列中取出任务然后执 ...

  8. 物联网学习笔记三:物联网网关协议比较:MQTT 和 Modbus

    物联网学习笔记三:物联网网关协议比较:MQTT 和 Modbus 物联网 (IoT) 不只是新技术,还是与旧技术的集成,其关键在于通信.可用的通信方法各不相同,但是,各种不同的协议在将海量“事物”连接 ...

  9. 官网英文版学习——RabbitMQ学习笔记(十)RabbitMQ集群

    在第二节我们进行了RabbitMQ的安装,现在我们就RabbitMQ进行集群的搭建进行学习,参考官网地址是:http://www.rabbitmq.com/clustering.html 首先我们来看 ...

随机推荐

  1. ssh无法连接到远端Ubuntu的解决方法

    近日,饱受无法远程登录到新安装在VMWare上的Ubuntu虚拟机,现在发现问题所在.故记录此问题的解决方式,以备后用. 一.远程登录虚拟机的准备: Ubuntu虚拟机的联网方式应该选择Bridged ...

  2. badboy提示脚本错误解决方法

    1.输入URL,提示脚本错误 解决办法:打开IE浏览器,工具->internet选项->高级,如图所示去掉禁用脚本调试 2.badboy内置浏览器,提示脚本错误解决办法 解决办法:badb ...

  3. layer获取弹出frame层数据

    通常,弹出层关闭之前,需要将部分数据传入父页面.这个时候怎么办呢? 通过success获取frame层的index. 然后通过cancel事件,获取子页面数据. 拿获取高德地图坐标为例: // 显示地 ...

  4. 框架-Eureka:初识 Eureka

    ylbtech-框架-Eureka:初识 Eureka 1.返回顶部 1. 1.1. http://localhost:2100/ 1.2. 2. Eureka - Last N events 3. ...

  5. 获取id 获取当前点击元素节点的任意 属性

    <a id="haveproces" onclick="fnProces(event)" dataid="{{x.id}}" clas ...

  6. 鸟哥的Linux私房菜笔记第六章(一)

    目录与路径 相对路径与绝对路径 上一章简单的提到绝对路径和相对路径 绝对路径:路径的写法一定是由根目录(/)写起的,例如:/home/user 这个目录 相对路径:路径的写法不是由根目录(/)写起,例 ...

  7. HBase编程 API入门系列之delete(管理端而言)(9)

    大家,若是看过我前期的这篇博客的话,则 HBase编程 API入门之delete(客户端而言) 就知道,在这篇博文里,我是在客户端里删除HBase表的. 这里,我带领大家,学习更高级的,因为,在开发中 ...

  8. Dalvik 堆内存管理与回收

    Dalvik虚拟机用来分配对象的堆划分为两部分,一部分叫做Active Heap,另一部分叫做Zygote Heap.下面基于管理机制来介绍为何分配为这两部分,以及堆内存的管理. 我们从Android ...

  9. Redux入门

    Redux入门 本文转载自:众成翻译 译者:miaoYu 链接:http://www.zcfy.cc/article/4728 原文:https://bumbu.github.io/simple-re ...

  10. 【MFC】如何在mfc窗口程序中调用控制台

    1.工程名为Zero,在CZeroDlg.cpp中加入头文件 #include “conio.h” : 2.在CZeroDlg::OnInitDialog() {…}函数中加入AllocConsole ...