前言:在这里我将用java来简单的实现rabbitMQ。下面我们带着下面问题来一步步的了解和学习rabbitMQ。

1:如果消费者连接中断,这期间我们应该怎么办

2:如何做到负载均衡

3:如何有效的将数据发送到相关的接收者?就是怎么样过滤

4:如何保证消费者收到完整正确的数据

5:如何让优先级高的接收者先收到数据

一:"Hello RabbitMQ"

下面有一幅图,其中P表示生产者,C表示消费者,红色部分为消息队列

 二:项目开始

2.1:首先引入rabbitMQ jar包

  1. <dependency>
  2. <groupId>com.rabbitmq</groupId>
  3. <artifactId>amqp-client</artifactId>
  4. <version>3.6.5</version>
  5. </dependency>

2.2:创建消费者Producer

  1. /**
  2. * 消息生成者
  3. */
  4. public class Producer {
  5. public final static String QUEUE_NAME="rabbitMQ.test";
  6.  
  7. public static void main(String[] args) throws IOException, TimeoutException {
  8. //创建连接工厂
  9. ConnectionFactory factory = new ConnectionFactory();
  10. //设置RabbitMQ相关信息
  11. factory.setHost("localhost");
  12. //factory.setUsername("lp");
  13. //factory.setPassword("");
  14. // factory.setPort(2088);
  15. //创建一个新的连接
  16. Connection connection = factory.newConnection();
  17. //创建一个通道
  18. Channel channel = connection.createChannel();
  19. // 声明一个队列 channel.queueDeclare(QUEUE_NAME, false, false, false, null);
  20. String message = "Hello RabbitMQ";
  21. //发送消息到队列中
  22. channel.basicPublish("", QUEUE_NAME, null, message.getBytes("UTF-8"));
  23. System.out.println("Producer Send +'" + message + "'");
  24. //关闭通道和连接
  25. channel.close();
  26. connection.close();
  27. }
  28. }

注1:queueDeclare第一个参数表示队列名称、第二个参数为是否持久化(true表示是,队列将在服务器重启时生存)、第三个参数为是否是独占队列(创建者可以使用的私有队列,断开后自动删除)、第四个参数为当所有消费者客户端连接断开时是否自动删除队列、第五个参数为队列的其他参数

注2:basicPublish第一个参数为交换机名称、第二个参数为队列映射的路由key、第三个参数为消息的其他属性、第四个参数为发送信息的主体

2.3:创建消费者

  1. public class Customer {
  2. private final static String QUEUE_NAME = "rabbitMQ.test";
  3.  
  4. public static void main(String[] args) throws IOException, TimeoutException {
  5. // 创建连接工厂
  6. ConnectionFactory factory = new ConnectionFactory();
  7. //设置RabbitMQ地址
  8. factory.setHost("localhost");
  9. //创建一个新的连接
  10. Connection connection = factory.newConnection();
  11. //创建一个通道
  12. Channel channel = connection.createChannel();
  13. //声明要关注的队列
  14. channel.queueDeclare(QUEUE_NAME, false, false, true, null);
  15. System.out.println("Customer Waiting Received messages");
  16. //DefaultConsumer类实现了Consumer接口,通过传入一个频道,
  17. // 告诉服务器我们需要那个频道的消息,如果频道中有消息,就会执行回调函数handleDelivery
  18. Consumer consumer = new DefaultConsumer(channel) {
  19. @Override
  20. public void handleDelivery(String consumerTag, Envelope envelope,
  21. AMQP.BasicProperties properties, byte[] body)
  22. throws IOException {
  23. String message = new String(body, "UTF-8");
  24. System.out.println("Customer Received '" + message + "'");
  25. }
  26. };
  27. //自动回复队列应答 -- RabbitMQ中的消息确认机制
  28. channel.basicConsume(QUEUE_NAME, true, consumer);
  29. }

前面代码我们可以看出和生成者一样的,后面的是获取生产者发送的信息,其中envelope主要存放生产者相关信息(比如交换机、路由key等)body是消息实体。

2.4:运行结果

生产者:

消费者:

 三:实现任务分发

工作队列

一个队列的优点就是很容易处理并行化的工作能力,但是如果我们积累了大量的工作,我们就需要更多的工作者来处理,这里就要采用分布机制了。

我们新创建一个生产者NewTask

  1. public class NewTask {
  2. private static final String TASK_QUEUE_NAME="task_queue";
  3. public static void main(String[] args) throws IOException, TimeoutException {
  4. ConnectionFactory factory=new ConnectionFactory();
  5. factory.setHost("localhost");
  6. Connection connection=factory.newConnection();
  7. Channel channel=connection.createChannel();
  8. channel.queueDeclare(TASK_QUEUE_NAME,true,false,false,null);
  9. //分发信息
  10. for (int i=0;i<10;i++){
  11. String message="Hello RabbitMQ"+i;
  12. channel.basicPublish("",TASK_QUEUE_NAME,
  13. MessageProperties.PERSISTENT_TEXT_PLAIN,message.getBytes());
  14. System.out.println("NewTask send '"+message+"'");
  15. }
  16. channel.close();
  17. connection.close();
  18. }
  19. }

然后创建2个工作者Work1和Work2代码一样

  1. public class Work1 {
  2. private static final String TASK_QUEUE_NAME = "task_queue";
  3.  
  4. public static void main(String[] args) throws IOException, TimeoutException {
  5. final ConnectionFactory factory = new ConnectionFactory();
  6. factory.setHost("localhost");
  7. Connection connection = factory.newConnection();
  8. final Channel channel = connection.createChannel();
  9.  
  10. channel.queueDeclare(TASK_QUEUE_NAME, true, false, false, null);
  11. System.out.println("Worker1 Waiting for messages");
  12.  
  13. //每次从队列获取的数量
  14. channel.basicQos(1);
  15.  
  16. final Consumer consumer = new DefaultConsumer(channel) {
  17. @Override
  18. public void handleDelivery(String consumerTag,
  19. Envelope envelope,
  20. AMQP.BasicProperties properties,
  21. byte[] body) throws IOException {
  22. String message = new String(body, "UTF-8");
  23. System.out.println("Worker1 Received '" + message + "'");
  24. try {
  25. throw new Exception();
  26. //doWork(message);
  27. }catch (Exception e){
  28. channel.abort();
  29. }finally {
  30. System.out.println("Worker1 Done");
  31. channel.basicAck(envelope.getDeliveryTag(),false);
  32. }
  33. }
  34. };
  35. boolean autoAck=false;
  36. //消息消费完成确认
  37. channel.basicConsume(TASK_QUEUE_NAME, autoAck, consumer);
  38. }
  39. private static void doWork(String task) {
  40. try {
  41. Thread.sleep(1000); // 暂停1秒钟
  42. } catch (InterruptedException _ignored) {
  43. Thread.currentThread().interrupt();
  44. }
  45. }
  46. }

注:channel.basicQos(1);保证一次只分发一个 。autoAck是否自动回复,如果为true的话,每次生产者只要发送信息就会从内存中删除,那么如果消费者程序异常退出,那么就无法获取数据,我们当然是不希望出现这样的情况,所以才去手动回复,每当消费者收到并处理信息然后在通知生成者。最后从队列中删除这条信息。如果消费者异常退出,如果还有其他消费者,那么就会把队列中的消息发送给其他消费者,如果没有,等消费者启动时候再次发送。

关于上面我们遗留问题在下一篇继续讲解

rabbitMQ第二篇:java简单的实现RabbitMQ的更多相关文章

  1. 【RabbitMQ】 Java简单的实现RabbitMQ

    准备工作 1.安装RabbitMQ,参考[RabbitMQ] RabbitMQ安装 2.新建Java项目,引入RabbitMQ的Maven依赖 <dependency> <group ...

  2. Android JNI入门第二篇——Java参数类型与本地参数类型对照

    前面一篇通过简单的例子介绍了android中JNI的使用.这一篇从基础上了解一些Java参数类型与本地参数类型区别. 1)        java中的返回值void和JNI中的void是完全对应的哦! ...

  3. Java并发包下锁学习第二篇Java并发基础框架-队列同步器介绍

    Java并发包下锁学习第二篇队列同步器 还记得在第一篇文章中,讲到的locks包下的类结果图吗?如下图: ​ 从图中,我们可以看到AbstractQueuedSynchronizer这个类很重要(在本 ...

  4. 《安卓网络编程》之第二篇 java环境下网络通信的综合应用

    经过前面内容的学习,我们了解了Java技术中实现网络通信的基本知识.下面将通过一个具体视力的实现过程,讲解客户端和服务器端通信的流程. 服务器端的实现文件是 Server.java,代码如下: imp ...

  5. 第二篇——VC++简单随机四则运算

    目标:编写最简单的四则运算,类似A+B=C: 想法:建立一个Win32控制台应用程序,A和B用随机数表示,运算符号用0~3的数字对应,然后计算并输出即可: 具体过程: 利用函数rand(),返回一个0 ...

  6. RabbitMQ学习总结 第二篇:快速入门HelloWorld

    目录 RabbitMQ学习总结 第一篇:理论篇 RabbitMQ学习总结 第二篇:快速入门HelloWorld RabbitMQ学习总结 第三篇:工作队列Work Queue RabbitMQ学习总结 ...

  7. RabbitMQ六种队列模式-简单队列模式

    前言 RabbitMQ六种队列模式-简单队列 [本文]RabbitMQ六种队列模式-工作队列RabbitMQ六种队列模式-发布订阅RabbitMQ六种队列模式-路由模式RabbitMQ六种队列模式-主 ...

  8. Java面试题(RabbitMQ篇)

    RabbitMQ 135. rabbitmq 的使用场景有哪些? ①. 跨系统的异步通信,所有需要异步交互的地方都可以使用消息队列.就像我们除了打电话(同步)以外,还需要发短信,发电子邮件(异步)的通 ...

  9. Python之路【第九篇】:Python操作 RabbitMQ、Redis、Memcache、SQLAlchemy

    Python之路[第九篇]:Python操作 RabbitMQ.Redis.Memcache.SQLAlchemy   Memcached Memcached 是一个高性能的分布式内存对象缓存系统,用 ...

随机推荐

  1. Spring 配置 Spring JPA 发生错误的解决方法

    今天在项目的applicationContext.xml中配JPA时 <?xml version="1.0" encoding="UTF-8"?> ...

  2. Python操控mysql数据库

    导入库——MySQLdb 1.链接MySQL conn=MySQL.connect(host=',port=3306) 此处主机用ip地址,不能使用'localhost',不然会显示链接错误. 2.选 ...

  3. Android图片复制

    public void saveImage2Phone(SlideShowImage image){ String imagePath; if(Environment.getExternalStora ...

  4. 这几天对Redis的初探,写一个阶段性的东西

    原来基于wcf写了一个交互框架,其中自定义了一个session队列,用于保存客户端连接的一些信息. 这几天在想如何将这个wcf框架做负载均衡,于是将session队列拆分出来,用一个共享的内存进行处理 ...

  5. 使用Excel对象模型在Excel单元格中设置不同的字体

    效果是这样的: 首先找到这个单元格或区域Range cell,然后代码: ((Range)cell). Characters[, ].Font.Color = Color.Blue; ((Range) ...

  6. FMX下Edit只能输入数字

    procedure TForm1.Edit1KeyDown(Sender: TObject; var Key: Word; var KeyChar: Char; Shift: TShiftState) ...

  7. CentOS7 服务器 JDK+TOMCAT+MYSQL+redis 安装日志

    防火墙配置(参考 CentOS7安装iptables防火墙) 检查是否安装iptables #先检查是否安装了iptables service iptables status #安装iptables ...

  8. 6.用CXF编写基于Spring的WebService

    首先是服务器端: //实体类 public class Weather { private String region;//区域编码 private String regionName;//区域名称 ...

  9. java 并发性和多线程 -- 读感 (一 线程的基本概念部分)

    1.目录略览      线程的基本概念:介绍线程的优点,代价,并发编程的模型.如何创建运行java 线程.      线程间通讯的机制:竞态条件与临界区,线程安全和共享资源与不可变性.java内存模型 ...

  10. android 如何结束一个线程

    总结: 1 不推荐直接调用onstop()强制结束,,因为不安全 2 run()比较短暂,执行完毕会自动停止 3 在run()设置一个flag标识,满足条件才执行; 4 通过sleep()捕获异常,在 ...