• Java 访问RabbitMQ


    一、概述

    前面学过ActiveMQ。ActiveMQ主要是实现的JMS规范,而RabbitMQ就是AMQP的一个具体实现。

    RabbitMQ里面有几个概念:生产者、消费者、消息、交换器、路由键、队列、绑定、虚拟主机

    1.生产者角度

      生产者产生数据,然后根据指定交换器和路由键将数据发送到消息队列RabbitMQ。为了保证交换器的存在,我们每次在初始化生产者的时候都要尝试去创建一个交换器。

      交换器总共有4种类型:

    1. direct 路由键完全匹配
    2. fanout 消息广播,将忽略路由键
    3. topic 通过“*”和“#”的通配符进行绑定。注意:”.”将路由键分为了几个标识符,“*”匹配1个,“#”匹配一个或多个
    4. headers 和direct类似,很少使用

    2.消费者角度

      消费者主要就是获取并消费数据,因此需要创建一个队列,同时需要创建一个交换器(交换器在消费者和生产者都可以创建),然后将队列和交换器通过路由键进行绑定。最后就可以根据队列进行数据的消费了。

    二、Java代码

    1.pom.xml

    <dependency>
        <groupId>com.rabbitmq</groupId>
        <artifactId>amqp-client</artifactId>
        <version>4.0.2</version>
        <exclusions>
            <exclusion>
                <groupId>org.slf4j</groupId>
                <artifactId>slf4j-api</artifactId>
            </exclusion>
        </exclusions>
    </dependency>

    2.生产者代码

    package cn.duanjt;
    
    import java.io.IOException;
    import java.util.concurrent.TimeoutException;
    
    import com.rabbitmq.client.BuiltinExchangeType;
    import com.rabbitmq.client.Channel;
    import com.rabbitmq.client.Connection;
    import com.rabbitmq.client.ConnectionFactory;
    
    /**
     * 生产者
     * @author 段江涛
     * @date 2018-11-30
     */
    public class Productor {
        public static void main(String[] args) throws IOException, TimeoutException {
            String ROUTE_KEY = "rabbitmq-duanjt";// 路由键名称
            String EXCHANGE_NAME = "exchange-duanjt";// 交换器名称
    
            ConnectionFactory factory = new ConnectionFactory();
            factory.setHost("192.168.23.24");
            factory.setPort(5672);
            factory.setUsername("admin");
            factory.setPassword("admin");
            factory.setVirtualHost("/");//虚拟主机,可通过控制台查看
    
            //创建连接和信道
            Connection conn = factory.newConnection();
            Channel channel = conn.createChannel();
    
            // 创建一个交换器,参数为:交互器名称和交换器类型
            // 注意:其实这个交换器只需要声明一次就可以,但是由于无法保证交换器已经存在了,所以我们每次都要声明
            channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT);
    
            for (int i = 0; i < 5; i++) {
                String msg = "Hello world.I love you forever ===>" + i;
                // 发布消息,需要参数:交换器,路由键。最后一个参数为消息内容
                // 注意:RabbitMQ的消息类型只有一种,那就是byte[]
                channel.basicPublish(EXCHANGE_NAME, ROUTE_KEY, null, msg.getBytes("utf-8"));
    
                System.out.println("send:" + msg);
            }
    
            //关闭信道和连接
            channel.close();
            conn.close();
        }
    }

    3.消费者代码

    package cn.duanjt;
    
    import java.io.IOException;
    import java.util.concurrent.TimeoutException;
    
    import com.rabbitmq.client.Channel;
    import com.rabbitmq.client.Connection;
    import com.rabbitmq.client.ConnectionFactory;
    import com.rabbitmq.client.DefaultConsumer;
    import com.rabbitmq.client.Envelope;
    import com.rabbitmq.client.AMQP.BasicProperties;
    import com.rabbitmq.client.BuiltinExchangeType;
    
    public class Consumer {
        public static void main(String[] args) throws IOException, TimeoutException {
            String QUEUE_NAME = "queue-duanjt";// 队列名称
            String ROUTE_KEY = "rabbitmq-duanjt";// 路由键名称
            String EXCHANGE_NAME = "exchange-duanjt";// 交换器名称
    
            ConnectionFactory factory = new ConnectionFactory();
            factory.setHost("192.168.23.24");
            factory.setPort(5672);
            factory.setUsername("admin");
            factory.setPassword("admin");
            factory.setVirtualHost("/");
    
            Connection conn = factory.newConnection();
            Channel channel = conn.createChannel();
            // 创建一个队列
            channel.queueDeclare(QUEUE_NAME, false, false, false, null);
            // 创建交换器
            channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT);
            // 将队列和交换器通过路由键进行绑定
            channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, ROUTE_KEY);
    
            //开始消费,第二个参数表示自动确认
            channel.basicConsume(QUEUE_NAME, true, new DefaultConsumer(channel) {
                // 当消息到达时执行回调方法
                @Override
                public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties, byte[] body) throws IOException {
                    String message = new String(body, "utf-8");
                    System.out.println("[Receive]:" + message);
                }
            });
    
        }
    }

      

    注意:

    1.为了保证交换器的存在,所以消费者和生产者都要创建,因为不知道是消费者先启动还是生产者先启动

    2.可以通过http://ip:15672 查看交换器、路由键和队列之间的关系

    3.一个连接(Connection)可以创建多个信道(Channel)。每个信道也可以在独立的一个线程里面

    4.一个队列可以有多个消费者,这种情况下,消息将在消费者之间进行轮询

  • 相关阅读:
    【jQuery插件】使用cropper实现简单的头像裁剪并上传
    Hibernate与 MyBatis的比较
    ES引擎原理介绍
    ES中的分词器研究
    服务转发Feign
    hytrix的python实现
    Consul的服务注册与发现-python版本
    setupTools工具新建项目
    docker系统学习
    NLP关键技术
  • 原文地址:https://www.cnblogs.com/duanjt/p/10057330.html
Copyright © 2020-2023  润新知