学习RabbitMQ这篇就够了快速入门上手(超详细)

2.3入门案例


2.3.1环境搭建

  • 构建环境:test_rabbitmq


<dependencies>

    <dependency>

        <groupId>org.springframework.boot</groupId>

        <artifactId>spring-boot-starter-amqp</artifactId>

    </dependency>

</dependencies>

2.3.2生产者


ackage com.czxy.xuecheng.rabbitmq;



import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.ConnectionFactory;



import java.io.IOException;

import java.util.concurrent.TimeoutException;



/**

 * Created by liangtong.

 */

public class Producer01 {

    //队列名称

    private static final String QUEUE = "helloworld";

    

    public static void main(String[] args) throws IOException, TimeoutException {

        Connection connection = null;

        Channel channel = null;

        try {

            ConnectionFactory factory = new ConnectionFactory();

            factory.setHost("localhost");

            factory.setPort(5672);

            factory.setUsername("guest");

            factory.setPassword("guest");

            factory.setVirtualHost("/");//rabbitmq默认虚拟机名称为“/”,虚拟机相当于一个独立的mq服务器

            // 创建与RabbitMQ服务的TCP连接

            connection = factory.newConnection();

            //创建与Exchange的通道,每个连接可以创建多个通道,每个通道代表一个会话任务

            channel = connection.createChannel();

            /**

             * 声明队列,如果Rabbit中没有此队列将自动创建

             * param1:队列名称

             * param2:是否持久化

             * param3:队列是否独占此连接

             * param4:队列不再使用时是否自动删除此队列

             * param5:队列参数

             */

            channel.queueDeclare(QUEUE, true, false, false, null);

            String message = "helloworld小明"+System.currentTimeMillis();

            /**

             * 消息发布方法

             * param1:Exchange的名称,如果没有指定,则使用Default Exchange

             * param2:routingKey,消息的路由Key,是用于Exchange(交换机)将消息转发到指定的消息队列

             * param3:消息包含的属性

             * param4:消息体

             */

            /**

             * 这里没有指定交换机,消息将发送给默认交换机,每个队列也会绑定那个默认的交换机,但是不能显

             示绑定或解除绑定

             * 默认的交换机,routingKey等于队列名称

             */

            channel.basicPublish("", QUEUE, null, message.getBytes());

            System.out.println("Send Message is:'" + message + "'");

        } catch(Exception ex) {

            ex.printStackTrace();

        } finally {

            if(channel != null) {

                channel.close();

            }

            if(connection != null) {

                connection.close();

            }

        }

    }



}

2.3.3消费者


package com.czxy.xuecheng.rabbitmq;



import com.rabbitmq.client.*;



import java.io.IOException;

import java.util.concurrent.TimeoutException;



/**

 * Created by liangtong.

 */

public class Consumer01 {



    private static final String QUEUE = "helloworld";

    public static void main(String[] args) throws IOException, TimeoutException {

        ConnectionFactory factory = new ConnectionFactory();

        //设置MabbitMQ所在服务器的ip和端口

        factory.setHost("127.0.0.1");

        factory.setPort(5672);

        Connection connection = factory.newConnection();

        Channel channel = connection.createChannel();

        //声明队列

        channel.queueDeclare(QUEUE, true, false, false, null);

        //定义消费方法

        DefaultConsumer consumer = new DefaultConsumer(channel) {

            /**

             * 消费者接收消息调用此方法

             * @param consumerTag 消费者的标签,在channel.basicConsume()去指定

             * @param envelope 消息包的内容,可从中获取消息id,消息routingkey,交换机,消息和重传标志(收到消息失败后是否需要重新发送)

             * @param properties

             * @param body

             * @throws IOException

             */

            @Override

            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {

                //交换机

                String exchange = envelope.getExchange();

                //路由key

                String routingKey = envelope.getRoutingKey();

                //消息id

                long deliveryTag = envelope.getDeliveryTag();

                //消息内容

                String msg = new String(body,"utf-8");

                System.out.println("receive message.." + msg);

            }

        };

        /**

        * 监听队列String queue, boolean autoAck,Consumer callback

        * 参数明细

        * 1、队列名称

        * 2、是否自动回复,设置为true为表示消息接收到自动向mq回复接收到了,mq接收到回复会删除消息,设置为false则需要手动回复

        * 3、消费消息的方法,消费者接收到消息后调用此方法

        */

        channel.basicConsume(QUEUE, true, consumer);

    }

}

2.4工作模式


RabbitMQ有以下几种工作模式 :

1、Work queues

2、Publish/Subscribe:发布订阅模式

3、Routing:路由模式

4、Topics

5、Header

6、RPC

2.4.1Word queues

work queues与入门程序相比,多了一个消费端,两个消费端共同消费同一个队列中的消息。

  1. * 测试:

1、使用入门程序,启动多个消费者。

2、生产者发送多个消息。

  1. 结果:

1、一条消息只会被一个消费者接收;

2、rabbit采用轮询的方式将消息是平均发送给消费者的;

3、消费者在处理完某条消息后,才会收到下一条消息。

2.4.2Publish/Subscribe

发布订阅模式:

1、每个消费者监听自己的队列。

2、生产者将消息发给broker,由交换机将消息转发到绑定此交换机的每个队列,每个绑定交换机的队列都将接收到消息

  1. 需求:用户通知,当用户充值成功或转账完成系统通知用户,通知方式有短信、邮件多种方法 。

  2. 1)生产者

声明inform_Exchange_fanout交换机。

声明两个队列并且绑定到此交换机,绑定时不需要指定routingkey

发送消息时不需要指定routingkey


package com.czxy.xuecheng.rabbitmq.demo02;



import com.rabbitmq.client.BuiltinExchangeType;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.ConnectionFactory;



import java.io.IOException;



/**

 * Created by liangtong.

 */

public class Producer02_publish {



    //队列名称

    private static final String QUEUE_INFORM_EMAIL = "queue_inform_email";

    private static final String QUEUE_INFORM_SMS = "queue_inform_sms";

    private static final String EXCHANGE_FANOUT_INFORM = "inform_exchange_fanout";



    public static void main(String[] args) {

        Connection connection = null;

        Channel channel = null;

        try {

            //创建一个与MQ的连接

            ConnectionFactory factory = new ConnectionFactory();

            factory.setHost("127.0.0.1");

            factory.setPort(5672);

            factory.setUsername("guest");

            factory.setPassword("guest");

            factory.setVirtualHost("/");//rabbitmq默认虚拟机名称为“/”,虚拟机相当于一个独立的mq服务器



            //创建一个连接

            connection = factory.newConnection();

            //创建与交换机的通道,每个通道代表一个会话

            channel = connection.createChannel();

            //声明交换机 String exchange, BuiltinExchangeType type

            /**

             * 参数明细

             * 1、交换机名称

             * 2、交换机类型,fanout、topic、direct、headers

             *  fanout:广播模式需要生产者消费者绑定相同的Exchange

             *  topic:支持模糊匹配的广播模式以点分隔,*表示一个单词,#表示任意数量(零个或多个)单词

             *  direct :需要生产者和消费者绑定相同的Exchange和routing key

             *  headers:根据生产者和消费者的header中信息进行匹配性能较差 ,x-match [all 匹配所有/any 任意一个]。

             */

            channel.exchangeDeclare(EXCHANGE_FANOUT_INFORM, BuiltinExchangeType.FANOUT);

            //声明队列

            // (String queue, boolean durable, boolean exclusive, boolean autoDelete, Map<String, Object> arguments)

            /**

             * 参数明细:

             * 1、队列名称

             * 2、是否持久化

             * 3、是否独占此队列

             * 4、队列不用是否自动删除

             * 5、参数

             */

            channel.queueDeclare(QUEUE_INFORM_EMAIL, true, false, false, null);

            channel.queueDeclare(QUEUE_INFORM_SMS, true, false, false, null);

            //交换机和队列绑定String queue, String exchange, String routingKey

            /**

             * 参数明细

             * 1、队列名称

             * 2、交换机名称

             * 3、路由key

             */

            channel.queueBind(QUEUE_INFORM_EMAIL, EXCHANGE_FANOUT_INFORM, "");

            channel.queueBind(QUEUE_INFORM_SMS, EXCHANGE_FANOUT_INFORM, "");

            //发送消息

            for (int i = 0; i < 10; i++) {

                String message = "inform to user" + i;

                //向交换机发送消息 String exchange, String routingKey, BasicProperties props, byte[] body

                /**

                 * 参数明细

                 * 1、交换机名称,不指令使用默认交换机名称 Default Exchange

                 * 2、routingKey(路由key),根据key名称将消息转发到具体的队列,这里填写队列名称表示消息将发到此队列

                 * 3、消息属性

                 * 4、消息内容

                 */

                channel.basicPublish(EXCHANGE_FANOUT_INFORM, "", null, message.getBytes());

                System.out.println("Send Message is:'" + message + "'");

            }



        } catch (Exception e) {

            e.printStackTrace();

        } finally {

            if (channel != null) {

                try {

                    channel.close();

                } catch (Exception e) {

                    e.printStackTrace();

                }

            }

            if (connection != null) {

                try {

                    connection.close();

                } catch (IOException e) {

                    e.printStackTrace();

                }

            }



        }

    }

}

  1. 2)邮件发送消费者

package com.czxy.xuecheng.rabbitmq.demo02;



import com.rabbitmq.client.*;



import java.io.IOException;



/**

 * Created by liangtong.

 */

public class Consumer02_subscribe_email {

    //队列名称

    private static final String QUEUE_INFORM_EMAIL = "queue_inform_email";

    private static final String EXCHANGE_FANOUT_INFORM = "inform_exchange_fanout";



    public static void main(String[] args) throws Exception {

        //创建一个与MQ的连接

        ConnectionFactory factory = new ConnectionFactory();

        factory.setHost("127.0.0.1");

        factory.setPort(5672);

        factory.setUsername("guest");

        factory.setPassword("guest");

        factory.setVirtualHost("/");//rabbitmq默认虚拟机名称为“/”,虚拟机相当于一个独立的mq服务器



        //创建一个连接

        Connection connection = factory.newConnection();

        //创建与交换机的通道,每个通道代表一个会话

        Channel channel = connection.createChannel();

        //声明交换机 String exchange, BuiltinExchangeType type

        /**

         * 参数明细

         * 1、交换机名称

         * 2、交换机类型,fanout、topic、direct、headers

         */

        channel.exchangeDeclare(EXCHANGE_FANOUT_INFORM, BuiltinExchangeType.FANOUT);

        //声明队列

        //channel.queueDeclare(String queue, boolean durable, boolean exclusive, boolean autoDelete, Map<String, Object> arguments)

        /**

         * 参数明细:

         * 1、队列名称

         * 2、是否持久化

         * 3、是否独占此队列

         * 4、队列不用是否自动删除

         * 5、参数

         */

        channel.queueDeclare(QUEUE_INFORM_EMAIL, true, false, false, null);

        //交换机和队列绑定String queue, String exchange, String routingKey

        /**

         * 参数明细

         * 1、队列名称

         * 2、交换机名称

         * 3、路由key

         */

        channel.queueBind(QUEUE_INFORM_EMAIL, EXCHANGE_FANOUT_INFORM, "");

        //定义消费方法

        DefaultConsumer defaultConsumer = new DefaultConsumer(channel) {

            @Override

            public void handleDelivery(String consumerTag, Envelope envelope,

                                       AMQP.BasicProperties properties, byte[] body) throws IOException {

                long deliveryTag = envelope.getDeliveryTag();

                String exchange = envelope.getExchange();

                //消息内容

                String message = new String(body, "UTF-8");

                System.out.println(message);

            }

        };

        /**

         * 监听队列String queue, boolean autoAck,Consumer callback

         * 参数明细

         * 1、队列名称

         * 2、是否自动回复,设置为true为表示消息接收到自动向mq回复接收到了,mq接收到回复会删除消息,设置

         为false则需要手动回复

         * 3、消费消息的方法,消费者接收到消息后调用此方法

         */

        channel.basicConsume(QUEUE_INFORM_EMAIL, true, defaultConsumer);





    }

}

  1. 3)短信发送消费者

package com.czxy.xuecheng.rabbitmq.demo02;



import com.rabbitmq.client.*;



import java.io.IOException;



/**

 * Created by liangtong.

 */

public class Consumer02_subscribe_sms {

    //队列名称

    private static final String QUEUE_INFORM_SMS = "queue_inform_sms";

    private static final String EXCHANGE_FANOUT_INFORM = "inform_exchange_fanout";



    public static void main(String[] args) throws Exception {

        //创建一个与MQ的连接

        ConnectionFactory factory = new ConnectionFactory();

        factory.setHost("127.0.0.1");

        factory.setPort(5672);

        factory.setUsername("guest");

        factory.setPassword("guest");

        //rabbitmq默认虚拟机名称为“/”,虚拟机相当于一个独立的mq服务器

        factory.setVirtualHost("/");



        //创建一个连接

        Connection connection = factory.newConnection();

        //创建与交换机的通道,每个通道代表一个会话

        Channel channel = connection.createChannel();

        //声明交换机 String exchange, BuiltinExchangeType type

        /**

         * 参数明细

         * 1、交换机名称

         * 2、交换机类型,fanout、topic、direct、headers

         */

        channel.exchangeDeclare(EXCHANGE_FANOUT_INFORM, BuiltinExchangeType.FANOUT);

        //声明队列

        //channel.queueDeclare(String queue, boolean durable, boolean exclusive, boolean autoDelete, Map<String, Object> arguments)

        /**

         * 参数明细:

         * 1、队列名称

         * 2、是否持久化

         * 3、是否独占此队列

         * 4、队列不用是否自动删除

         * 5、参数

         */

        channel.queueDeclare(QUEUE_INFORM_SMS, true, false, false, null);

        //交换机和队列绑定String queue, String exchange, String routingKey

        /**

         * 参数明细

         * 1、队列名称

         * 2、交换机名称

         * 3、路由key

         */

        channel.queueBind(QUEUE_INFORM_SMS, EXCHANGE_FANOUT_INFORM, "");

        //定义消费方法

        DefaultConsumer defaultConsumer = new DefaultConsumer(channel) {

            @Override

            public void handleDelivery(String consumerTag, Envelope envelope,

                                       AMQP.BasicProperties properties, byte[] body) throws IOException {

                long deliveryTag = envelope.getDeliveryTag();

                String exchange = envelope.getExchange();

                //消息内容

                String message = new String(body, "UTF-8");

                System.out.println(message);

            }

        };

        /**

         * 监听队列String queue, boolean autoAck,Consumer callback

         * 参数明细

         * 1、队列名称

         * 2、是否自动回复,设置为true为表示消息接收到自动向mq回复接收到了,mq接收到回复会删除消息,设置

         为false则需要手动回复

         * 3、消费消息的方法,消费者接收到消息后调用此方法

         */

        channel.basicConsume(QUEUE_INFORM_SMS, true, defaultConsumer);





    }

}

2.4.3Routing

路由模式:

1、每个消费者监听自己的队列,并且设置routingkey。

2、生产者将消息发给交换机,由交换机根据routingkey来转发消息到指定的队列。

  1. 1)生产者

声明inform_exchange_routing交换机。

声明两个队列并且绑定到此交换机,绑定时需要指定routingkey

发送消息时需要指定routingkey


package com.czxy.xuecheng.rabbitmq.demo03;



import com.rabbitmq.client.BuiltinExchangeType;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.ConnectionFactory;



import java.io.IOException;



/**

 * Created by liangtong.

 */

public class Producer03_routing {

    //队列名称

    private static final String QUEUE_INFORM_EMAIL = "queue_inform_email";

    private static final String QUEUE_INFORM_SMS = "queue_inform_sms";

    private static final String EXCHANGE_ROUTING_INFORM = "inform_exchange_routing";



    public static void main(String[] args) {

        Connection connection = null;

        Channel channel = null;

        try {

            //创建一个与MQ的连接

            ConnectionFactory factory = new ConnectionFactory();

            factory.setHost("127.0.0.1");

            factory.setPort(5672);

            factory.setUsername("guest");

            factory.setPassword("guest");

            //rabbitmq默认虚拟机名称为“/”,虚拟机相当于一个独立的mq服务器

            factory.setVirtualHost("/");



            //创建一个连接

            connection = factory.newConnection();

            //创建与交换机的通道,每个通道代表一个会话

            channel = connection.createChannel();

            //声明交换机 String exchange, BuiltinExchangeType type

            /**

             * 参数明细

             * 1、交换机名称

             * 2、交换机类型,fanout、topic、direct、headers

             */

            channel.exchangeDeclare(EXCHANGE_ROUTING_INFORM, BuiltinExchangeType.DIRECT);

            //声明队列

            //channel.queueDeclare(String queue, boolean durable, boolean exclusive, boolean autoDelete, Map < String, Object > arguments)

            /**

             * 参数明细:

             * 1、队列名称

             * 2、是否持久化

             * 3、是否独占此队列

             * 4、队列不用是否自动删除

             * 5、参数

             */

            channel.queueDeclare(QUEUE_INFORM_EMAIL, true, false, false, null);

            channel.queueDeclare(QUEUE_INFORM_SMS, true, false, false, null);

            //交换机和队列绑定String queue, String exchange, String routingKey

            /**

             * 参数明细

             * 1、队列名称

             * 2、交换机名称

             * 3、路由key

             */

            channel.queueBind(QUEUE_INFORM_EMAIL, EXCHANGE_ROUTING_INFORM, QUEUE_INFORM_EMAIL);

            channel.queueBind(QUEUE_INFORM_SMS, EXCHANGE_ROUTING_INFORM, QUEUE_INFORM_SMS);

            //发送邮件消息

            for (int i = 0; i < 10; i++) {

                String message = "email inform to user" + i;

                //向交换机发送消息 String exchange, String routingKey, BasicProperties props,byte[] body

                /**

                 * 参数明细

                 * 1、交换机名称,不指令使用默认交换机名称 Default Exchange

                 * 2、routingKey(路由key),根据key名称将消息转发到具体的队列,这里填写队列名称表示消

                 息将发到此队列

                 * 3、消息属性

                 * 4、消息内容

                 */

                channel.basicPublish(EXCHANGE_ROUTING_INFORM, QUEUE_INFORM_EMAIL, null, message.getBytes());

                System.out.println("Send Message is:'" + message + "'");

            }

            //发送短信消息

            for (int i = 0; i < 10; i++) {

                String message = "sms inform to user" + i;

                //向交换机发送消息 String exchange, String routingKey, BasicProperties props,byte[] body

                channel.basicPublish(EXCHANGE_ROUTING_INFORM, QUEUE_INFORM_SMS, null, message.getBytes());

                System.out.println("Send Message is:'" + message + "'");

            }



        } catch (Exception e) {

            e.printStackTrace();

        }  finally {

            if (channel != null) {

                try {

                    channel.close();

                } catch (Exception e) {

                    e.printStackTrace();

                }

            }

            if (connection != null) {

                try {

                    connection.close();

                } catch (IOException e) {

                    e.printStackTrace();

                }

            }



        }





    }

}

  1. 2)邮件发送消费者

package com.czxy.xuecheng.rabbitmq.demo03;



import com.rabbitmq.client.*;



import java.io.IOException;



/**

 * Created by liangtong.

 */

public class Consumer03_routing_email {

    //队列名称

    private static final String QUEUE_INFORM_EMAIL = "queue_inform_email";

    private static final String EXCHANGE_ROUTING_INFORM = "inform_exchange_routing";



    public static void main(String[] args) throws Exception {

        //创建一个与MQ的连接

        ConnectionFactory factory = new ConnectionFactory();

        factory.setHost("127.0.0.1");

        factory.setPort(5672);

        factory.setUsername("guest");

        factory.setPassword("guest");

        //rabbitmq默认虚拟机名称为“/”,虚拟机相当于一个独立的mq服务器

        factory.setVirtualHost("/");



        //创建一个连接

        Connection connection = factory.newConnection();

        //创建与交换机的通道,每个通道代表一个会话

        Channel channel = connection.createChannel();

        //声明交换机 String exchange, BuiltinExchangeType type

        /**

         * 参数明细

         * 1、交换机名称

         * 2、交换机类型,fanout、topic、direct、headers

         */

        channel.exchangeDeclare(EXCHANGE_ROUTING_INFORM, BuiltinExchangeType.DIRECT);

        //声明队列

        //channel.queueDeclare(String queue, boolean durable, boolean exclusive, booleanautoDelete, Map<String, Object> arguments)

        /**

         * 参数明细:

         * 1、队列名称

         * 2、是否持久化

         * 3、是否独占此队列

         * 4、队列不用是否自动删除

         * 5、参数

         */

        channel.queueDeclare(QUEUE_INFORM_EMAIL, true, false, false, null);

        //交换机和队列绑定String queue, String exchange, String routingKey

        /**

         * 参数明细

         * 1、队列名称

         * 2、交换机名称

         * 3、路由key

         */

        channel.queueBind(QUEUE_INFORM_EMAIL, EXCHANGE_ROUTING_INFORM, QUEUE_INFORM_EMAIL);

        //定义消费方法

        DefaultConsumer defaultConsumer = new DefaultConsumer(channel) {

            @Override

            public void handleDelivery(String consumerTag, Envelope envelope,

                                       AMQP.BasicProperties properties, byte[] body) throws IOException {

                long deliveryTag = envelope.getDeliveryTag();

                String exchange = envelope.getExchange();

                //消息内容

                String message = new String(body, "UTF-8");

                System.out.println(message);

            }

        };

        /**

         * 监听队列String queue, boolean autoAck,Consumer callback

         * 参数明细

         * 1、队列名称

         * 2、是否自动回复,设置为true为表示消息接收到自动向mq回复接收到了,mq接收到回复会删除消息,设置

         为false则需要手动回复

         * 3、消费消息的方法,消费者接收到消息后调用此方法

         */

        channel.basicConsume(QUEUE_INFORM_EMAIL, true, defaultConsumer);





    }

}

  1. 3)短信发送消费者

参考邮件发送消费者的代码流程,编写短信通知的代码。


package com.czxy.xuecheng.rabbitmq.demo03;



import com.rabbitmq.client.*;



import java.io.IOException;



/**

 * Created by liangtong.

 */

public class Consumer03_routing_sms {

    //队列名称

    private static final String QUEUE_INFORM_SMS = "queue_inform_sms";

    private static final String EXCHANGE_ROUTING_INFORM = "inform_exchange_routing";



    public static void main(String[] args) throws Exception {

        //创建一个与MQ的连接

        ConnectionFactory factory = new ConnectionFactory();

        factory.setHost("127.0.0.1");

        factory.setPort(5672);

        factory.setUsername("guest");

        factory.setPassword("guest");

        //rabbitmq默认虚拟机名称为“/”,虚拟机相当于一个独立的mq服务器

        factory.setVirtualHost("/");



        //创建一个连接

        Connection connection = factory.newConnection();

        //创建与交换机的通道,每个通道代表一个会话

        Channel channel = connection.createChannel();

        //声明交换机 String exchange, BuiltinExchangeType type

        /**

         * 参数明细

         * 1、交换机名称

         * 2、交换机类型,fanout、topic、direct、headers

         */

        channel.exchangeDeclare(EXCHANGE_ROUTING_INFORM, BuiltinExchangeType.DIRECT);

        //声明队列

        //channel.queueDeclare(String queue, boolean durable, boolean exclusive, booleanautoDelete, Map<String, Object> arguments)

        /**

         * 参数明细:

         * 1、队列名称

         * 2、是否持久化

         * 3、是否独占此队列

         * 4、队列不用是否自动删除

         * 5、参数

         */

        channel.queueDeclare(QUEUE_INFORM_SMS, true, false, false, null);

        //交换机和队列绑定String queue, String exchange, String routingKey

        /**

         * 参数明细

         * 1、队列名称

         * 2、交换机名称

         * 3、路由key

         */

        channel.queueBind(QUEUE_INFORM_SMS, EXCHANGE_ROUTING_INFORM, QUEUE_INFORM_SMS);

        //定义消费方法

        DefaultConsumer defaultConsumer = new DefaultConsumer(channel) {

            @Override

            public void handleDelivery(String consumerTag, Envelope envelope,

                                       AMQP.BasicProperties properties, byte[] body) throws IOException {

                long deliveryTag = envelope.getDeliveryTag();

                String exchange = envelope.getExchange();

                //消息内容

                String message = new String(body, "UTF-8");

                System.out.println(message);

            }

        };

        /**

         * 监听队列String queue, boolean autoAck,Consumer callback

         * 参数明细

         * 1、队列名称

         * 2、是否自动回复,设置为true为表示消息接收到自动向mq回复接收到了,mq接收到回复会删除消息,设置

         为false则需要手动回复

         * 3、消费消息的方法,消费者接收到消息后调用此方法

         */

        channel.basicConsume(QUEUE_INFORM_SMS, true, defaultConsumer);



    }

}

2.4.4Topic

  1. 主题模式

1、每个消费者监听自己的队列,并且设置带统配符的routingkey。

2、生产者将消息发给broker,由交换机根据routingkey来转发消息到指定的队列。

  1. 需求:根据用户的通知设置去通知用户,设置接收Email的用户只接收Email,设置接收sms的用户只接收sms,设置两种通知类型都接收的则两种通知都有效。

  2. 1)生产者

队列绑定交换机指定通配符:

统配符规则:

中间以“.”分隔。

符号#可以匹配多个词,符号*可以匹配一个词语。


package com.czxy.xuecheng.rabbitmq.demo04;



import com.rabbitmq.client.BuiltinExchangeType;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.ConnectionFactory;



import java.io.IOException;



/**

 * Created by liangtong.

 */

public class Producer04_topic {

    //队列名称

    private static final String QUEUE_INFORM_EMAIL = "queue_inform_email";

    private static final String QUEUE_INFORM_SMS = "queue_inform_sms";

    private static final String EXCHANGE_TOPIC_INFORM = "inform_exchange_topic";



    public static void main(String[] args) {

        Connection connection = null;

        Channel channel = null;

        try {

            //创建一个与MQ的连接

            ConnectionFactory factory = new ConnectionFactory();

            factory.setHost("127.0.0.1");

            factory.setPort(5672);

            factory.setUsername("guest");

            factory.setPassword("guest");

            //rabbitmq默认虚拟机名称为“/”,虚拟机相当于一个独立的mq服务器

            factory.setVirtualHost("/");



            //创建一个连接

            connection = factory.newConnection();

            //创建与交换机的通道,每个通道代表一个会话

            channel = connection.createChannel();

            //声明交换机 String exchange, BuiltinExchangeType type

            /**

             * 参数明细

             * 1、交换机名称

             * 2、交换机类型,fanout、topic、direct、headers

             */

            channel.exchangeDeclare(EXCHANGE_TOPIC_INFORM, BuiltinExchangeType.TOPIC);

            //声明队列

            //channel.queueDeclare(String queue, boolean durable, boolean exclusive, boolean autoDelete, Map < String, Object > arguments)

            /**

             * 参数明细:

             * 1、队列名称

             * 2、是否持久化

             * 3、是否独占此队列

             * 4、队列不用是否自动删除

             * 5、参数

             */

            channel.queueDeclare(QUEUE_INFORM_EMAIL, true, false, false, null);

            channel.queueDeclare(QUEUE_INFORM_SMS, true, false, false, null);

            //交换机和队列绑定String queue, String exchange, String routingKey

            /**

             * 参数明细

             * 1、队列名称

             * 2、交换机名称

             * 3、路由key

             */

            channel.queueBind(QUEUE_INFORM_EMAIL, EXCHANGE_TOPIC_INFORM, "inform.#.email.#");

            channel.queueBind(QUEUE_INFORM_SMS, EXCHANGE_TOPIC_INFORM, "inform.#.sms.#");

            //发送邮件消息

            for (int i = 0; i < 10; i++) {

                String message = "email inform to user" + i;

                //向交换机发送消息 String exchange, String routingKey, BasicProperties props,byte[] body

                /**

                 * 参数明细

                 * 1、交换机名称,不指令使用默认交换机名称 Default Exchange

                 * 2、routingKey(路由key),根据key名称将消息转发到具体的队列,这里填写队列名称表示消

                 息将发到此队列

                 * 3、消息属性

                 * 4、消息内容

                 */

                channel.basicPublish(EXCHANGE_TOPIC_INFORM, "inform.email", null, message.getBytes());

                System.out.println("Send Message is:'" + message + "'");

            }

            //发送短信消息

            for (int i = 0; i < 10; i++) {

                String message = "sms inform to user" + i;

                //向交换机发送消息 String exchange, String routingKey, BasicProperties props,byte[] body

                channel.basicPublish(EXCHANGE_TOPIC_INFORM, "inform.sms", null, message.getBytes());

                System.out.println("Send Message is:'" + message + "'");

            }



            //发送邮件和短信消息

            for (int i = 0; i < 10; i++) {

                String message = "sms and email inform to user" + i;

                //向交换机发送消息 String exchange, String routingKey, BasicProperties props,byte[] body

                channel.basicPublish(EXCHANGE_TOPIC_INFORM, "inform.sms.email", null, message.getBytes());

                System.out.println("Send Message is:'" + message + "'");

            }



        } catch (Exception e) {

            e.printStackTrace();

        }  finally {

            if (channel != null) {

                try {

                    channel.close();

                } catch (Exception e) {

                    e.printStackTrace();

                }

            }

            if (connection != null) {


**Java面试核心知识点笔记**

其中囊括了JVM、锁、并发、Java反射、Spring原理、微服务、Zookeeper、数据库、数据结构等大量知识点。

![蚂蚁金服(Java研发岗),26岁小伙斩获三面,收获Offer定级P6](https://img-blog.csdnimg.cn/img_convert/d59350afbda582d75fb7f2e599eee5a6.webp?x-oss-process=image/format,png)

**Java中高级面试高频考点整理**

![蚂蚁金服(Java研发岗),26岁小伙斩获三面,收获Offer定级P6](https://img-blog.csdnimg.cn/img_convert/5e33c7ced1c6b01804c4b548f98da464.webp?x-oss-process=image/format,png)

![蚂蚁金服(Java研发岗),26岁小伙斩获三面,收获Offer定级P6](https://img-blog.csdnimg.cn/img_convert/4a025245e83faceaa1ade086f7f18445.webp?x-oss-process=image/format,png)

**最后分享Java进阶学习及面试必备的视频教学**

![蚂蚁金服(Java研发岗),26岁小伙斩获三面,收获Offer定级P6](https://img-blog.csdnimg.cn/img_convert/b36d5edfd46cf82646ad39392d1095d7.webp?x-oss-process=image/format,png)

        * 3、消息属性

                 * 4、消息内容

                 */

                channel.basicPublish(EXCHANGE_TOPIC_INFORM, "inform.email", null, message.getBytes());

                System.out.println("Send Message is:'" + message + "'");

            }

            //发送短信消息

            for (int i = 0; i < 10; i++) {

                String message = "sms inform to user" + i;

                //向交换机发送消息 String exchange, String routingKey, BasicProperties props,byte[] body

                channel.basicPublish(EXCHANGE_TOPIC_INFORM, "inform.sms", null, message.getBytes());

                System.out.println("Send Message is:'" + message + "'");

            }



            //发送邮件和短信消息

            for (int i = 0; i < 10; i++) {

                String message = "sms and email inform to user" + i;

                //向交换机发送消息 String exchange, String routingKey, BasicProperties props,byte[] body

                channel.basicPublish(EXCHANGE_TOPIC_INFORM, "inform.sms.email", null, message.getBytes());

                System.out.println("Send Message is:'" + message + "'");

            }



        } catch (Exception e) {

            e.printStackTrace();

        }  finally {

            if (channel != null) {

                try {

                    channel.close();

                } catch (Exception e) {

                    e.printStackTrace();

                }

            }

            if (connection != null) {


**Java面试核心知识点笔记**

其中囊括了JVM、锁、并发、Java反射、Spring原理、微服务、Zookeeper、数据库、数据结构等大量知识点。

[外链图片转存中...(img-e1M4pIOY-1720102139294)]

**Java中高级面试高频考点整理**

[外链图片转存中...(img-6POchm7H-1720102139295)]

[外链图片转存中...(img-5DSY9tlR-1720102139296)]

**最后分享Java进阶学习及面试必备的视频教学**

[外链图片转存中...(img-OA59RmYN-1720102139296)]

  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值