rabbitMQ第二篇:java简单的实现RabbitMQ

摘要:
下面我们带着下面问题来一步步的了解和学习rabbitMQ。

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

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

2:如何做到负载均衡

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

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

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

一:"Hello RabbitMQ"

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

rabbitMQ第二篇:java简单的实现RabbitMQ第1张

二:项目开始

2.1:首先引入rabbitMQ jar包

 <dependency>
            <groupId>com.rabbitmq</groupId>
            <artifactId>amqp-client</artifactId>
            <version>3.6.5</version>
 </dependency>

2.2:创建消费者Producer

/*** 消息生成者
 */
public classProducer {
    public final static String QUEUE_NAME="rabbitMQ.test";

    public static void main(String[] args) throwsIOException, TimeoutException {
        //创建连接工厂
        ConnectionFactory factory = newConnectionFactory();
        //设置RabbitMQ相关信息
        factory.setHost("localhost");
      //factory.setUsername("lp");
      //factory.setPassword("");
     //factory.setPort(2088);
        //创建一个新的连接
        Connection connection =factory.newConnection();
        //创建一个通道
        Channel channel =connection.createChannel();
        //声明一个队列        channel.queueDeclare(QUEUE_NAME, false, false, false, null);
        String message = "Hello RabbitMQ";
        //发送消息到队列中
        channel.basicPublish("", QUEUE_NAME, null, message.getBytes("UTF-8"));
        System.out.println("Producer Send +'" + message + "'");
        //关闭通道和连接
channel.close();
        connection.close();
    }
}

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

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

2.3:创建消费者

public classCustomer {
    private final static String QUEUE_NAME = "rabbitMQ.test";

    public static void main(String[] args) throwsIOException, TimeoutException {
        //创建连接工厂
        ConnectionFactory factory = newConnectionFactory();
        //设置RabbitMQ地址
        factory.setHost("localhost");
        //创建一个新的连接
        Connection connection =factory.newConnection();
        //创建一个通道
        Channel channel =connection.createChannel();
        //声明要关注的队列
        channel.queueDeclare(QUEUE_NAME, false, false, true, null);
        System.out.println("Customer Waiting Received messages");
        //DefaultConsumer类实现了Consumer接口,通过传入一个频道,
        //告诉服务器我们需要那个频道的消息,如果频道中有消息,就会执行回调函数handleDelivery
        Consumer consumer = newDefaultConsumer(channel) {
            @Override
            public voidhandleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body)
                    throwsIOException {
                String message = new String(body, "UTF-8");
                System.out.println("Customer Received '" + message + "'");
            }
        };
        //自动回复队列应答 -- RabbitMQ中的消息确认机制
        channel.basicConsume(QUEUE_NAME, true, consumer);
    }

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

2.4:运行结果

生产者:

rabbitMQ第二篇:java简单的实现RabbitMQ第2张

消费者:

rabbitMQ第二篇:java简单的实现RabbitMQ第3张

三:实现任务分发

工作队列

rabbitMQ第二篇:java简单的实现RabbitMQ第4张

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

我们新创建一个生产者NewTask

public classNewTask {
    private static final String TASK_QUEUE_NAME="task_queue";
    public static void main(String[] args) throwsIOException, TimeoutException {
        ConnectionFactory factory=newConnectionFactory();
        factory.setHost("localhost");
        Connection connection=factory.newConnection();
        Channel channel=connection.createChannel();
   channel.queueDeclare(TASK_QUEUE_NAME,true,false,false,null);
        //分发信息
        for (int i=0;i<10;i++){
            String message="Hello RabbitMQ"+i;
            channel.basicPublish("",TASK_QUEUE_NAME,
                    MessageProperties.PERSISTENT_TEXT_PLAIN,message.getBytes());
            System.out.println("NewTask send '"+message+"'");
        }
        channel.close();
        connection.close();
    }
}

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

public classWork1 {
    private static final String TASK_QUEUE_NAME = "task_queue";

    public static void main(String[] args) throwsIOException, TimeoutException {
        final ConnectionFactory factory = newConnectionFactory();
        factory.setHost("localhost");
        Connection connection =factory.newConnection();
        final Channel channel =connection.createChannel();

        channel.queueDeclare(TASK_QUEUE_NAME, true, false, false, null);
        System.out.println("Worker1  Waiting for messages");

        //每次从队列获取的数量
        channel.basicQos(1);

        final Consumer consumer = newDefaultConsumer(channel) {
            @Override
            public voidhandleDelivery(String consumerTag,
                                       Envelope envelope,
                                       AMQP.BasicProperties properties,
                                       byte[] body) throwsIOException {
                String message = new String(body, "UTF-8");
                System.out.println("Worker1  Received '" + message + "'");
                try{
                    throw  newException();
                    //doWork(message);
                }catch(Exception e){
                    channel.abort();
                }finally{
                    System.out.println("Worker1 Done");
                    channel.basicAck(envelope.getDeliveryTag(),false);
                }
            }
        };
        boolean autoAck=false;
        //消息消费完成确认
channel.basicConsume(TASK_QUEUE_NAME, autoAck, consumer);
    }
    private static voiddoWork(String task) {
        try{
            Thread.sleep(1000); //暂停1秒钟
        } catch(InterruptedException _ignored) {
            Thread.currentThread().interrupt();
        }
    }
}

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

rabbitMQ第二篇:java简单的实现RabbitMQ第5张

rabbitMQ第二篇:java简单的实现RabbitMQ第6张

rabbitMQ第二篇:java简单的实现RabbitMQ第7张

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

免责声明:文章转载自《rabbitMQ第二篇:java简单的实现RabbitMQ》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇ORA-00001: unique constraint (string.string) violated 违反唯一约束条件(.)MyBatis 映射文件详解(六)下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

java根据图片路径下载图片并保存到本地目录

importjava.io.File; importjava.io.FileOutputStream; importjava.io.InputStream; importjava.io.OutputStream; importjava.net.URL; importjava.net.URLConnection; publicclassDownloadIma...

libcurl HTTP POST请求向服务器发送json数据

转载:http://blog.csdn.net/dgyanyong/article/details/14166217 转载:http://blog.csdn.net/th_gsb/article/details/42810007 转载:http://www.cnblogs.com/yangxunpeng/articles/7040697.html 转载:h...

友盟官方文档

#  产品概述 关于分享和授权的sdk接口,我们在v4.4.0做出了巨大的改变,精简了接口调用的代码。并将分享授权,与评论等功能做出了隔离,使结构更加清晰。所以本版本的功能也只有分享与授权并无其他功能,请开发者注意。 注意:本文示例代码只有分享与授权功能,并对接口进行了调整,如需要原功能的开发者,请继续使用v4.3.0版本。 # 获取友盟Appkey 如果你...

C++(四十八) — string容器的基本操作

参考博客:https://blog.csdn.net/qq_37941471/article/details/82107077 https://www.cnblogs.com/danielStudy/p/7127564.html#top 1、声明一个字符串 标准库类型string表示可变长的字符序列,为了在程序中使用string类型,我们必须包含头文件:#...

mybatis-plus学习

mybatis-plus 一、简介 通过maven项目,springBoot构建学习,官方地址文档简介 | MyBatis-Plus (baomidou.com) 官方讲解,mybatis的优点: 无侵入:只做增强不做改变,引入它不会对现有工程产生影响,如丝般顺滑 损耗小:启动即会自动注入基本 CURD,性能基本无损耗,直接面向对象操作 强大的 CRUD...

Java 生成指定时间范围的随机时间、随机中文姓名、随机字符姓名、随机数

解决问题: Java生成指定时间范围的随机时间? Java生成随机中文姓名? Java生成随机字符姓名? Java生成随机数? 代码: import java.io.UnsupportedEncodingException; import java.text.ParseException; import java.text.SimpleDateFormat...