普通Maven项目使用RabbitMQ

一、简单模式

一个服务者对应一个消费者

  1. 创建BaseProject父maven项目,在父项目底下创建maven模块producer
  2. 导入相关依赖
<!-- rabbitmq依赖 -->
        <dependency>
            <groupId>com.rabbitmq</groupId>
            <artifactId>amqp-client</artifactId>
            <version>5.9.0</version>
        </dependency>

        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-log4j12</artifactId>
            <version>1.7.25</version>
        </dependency>

        <dependency>
            <groupId>org.apache.commons</groupId>
            <artifactId>commons-lang3</artifactId>
            <version>3.3.2</version>
        </dependency>

        <!-- 整合到spring项目需要导入此依赖 -->
        <dependency>
            <groupId>org.springframework.amqp</groupId>
            <artifactId>spring-rabbit</artifactId>
            <version>2.3.10</version>
        </dependency>
  1. 在resource目录下创建log4j.properties
log4j.rootLogger=DEBUG,A1 log4j.logger.com.taotao = DEBUG 
log4j.logger.org.mybatis = DEBUG
log4j.appender.A1=org.apache.log4j.ConsoleAppender
log4j.appender.A1.layout=org.apache.log4j.PatternLayout
log4j.appender.A1.layout.ConversionPattern=%-d{yyyy-MM-dd HH:mm:ss,SSS} [%t] [%c]-[%p] %m%n
  1. 创建RabbitMqUtil工具类
private static ConnectionFactory factory;

    public static Connection getConnection() throws IOException, TimeoutException {
        //创建mq的连接工厂对象
        factory = new ConnectionFactory();
        //设置连接rabbit的主机地址
        factory.setHost("主机ip");
        //设置连接端口
        factory.setPort(5672);
        //设置连接哪个虚拟主机
        factory.setVirtualHost("host1");
        //设置虚拟主机的用户名和密码
        factory.setUsername("zhangsan");
        factory.setPassword("123");
        Connection connection = factory.newConnection();
        return connection;
    }
  1. 服务提供者类发送消息

向队列queue1发送消息,如果该消息没被消费者消费,消息是不会消失的。

public class SendMsg {

    public static void main(String[] args) {

        String msg = "hello RabbitMq123!";

        Connection connection = null;

        try {
            //获取连接,相当于JDBC的获取数据库连接
            connection = MQUtil.getConnection();
            /**
             * 相当于JDBC的statement
             * 声明(创建)队列,如果存在就不创建,不存在就创建
             * 参数1 队列名,
             * 参数2 durable: 是否持久化, 默认是存放到内存中,如果rabbitmq重启会丢失,如果想重启之后还存在就要使队列持久化,保存到Erlang自带的Mnesia数据库中,当rabbitmq重启之后会读取该数据库
             * 参数3 exclusive:是否排外的,当当前链接关闭时,是否删除这个队列
             * 参数4 autoDelete:是否自动删除,当最后一个消费者断开连接之后队列是否自动被删除(无论队列中是否有数据)
             * 参数5 arguments: 参数
             * channel.queueDeclare("queue1", false, false, true, null);
             */
            Channel channel = connection.createChannel();
            /**
             * 四个参数:
             * 参数1 : 交换机名称,
             * 参数2 : 目标队列名称,
             * 参数3 : 设置当前这条消息的属性(比如设置超时时间)
             * 参数4 : 消息内容
             */
            channel.basicPublish("","queue1",null,msg.getBytes());
            System.out.println("发送--->" + msg);

            //关闭
            channel.close();
            connection.close();

        } catch (IOException e) {
            e.printStackTrace();
        } catch (TimeoutException e) {
            e.printStackTrace();
        }

    }
  1. 创建服务消费者模块consumer

导包、log4j配置、MQUtil工具类,与上述一样。

消费消息,消息消费过后,队列中的消息会清空。

//接受消息
public class ReceiveMsg {

    public static void main(String[] args) {

        Connection connection = null;

        //获取连接,相当于JDBC的获取数据库连接
        try {
            connection = MQUtil.getConnection();
            Channel channel = connection.createChannel();

            //声明要关注的队列
            //channel.queueDeclare("queue1", false, false, false, null);
            //DefaultConsumer类实现了Consumer接口,通过传入一个频道,
            // 告诉服务器我们需要那个频道的消息,如果频道中有消息,就会执行回调函数handleDelivery
            Consumer consumer = new DefaultConsumer(channel) {
                public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                    String message = new String(body, "UTF-8");
                    System.out.println("消费者消费消息:'" + message + "'");
                }
            };
            //自动回复队列应答 -- RabbitMQ中的消息确认机制
            channel.basicConsume("queue1", true, consumer);

        } catch (IOException e) {
            e.printStackTrace();
        } catch (TimeoutException e) {
            e.printStackTrace();
        }

    }
}

总结:发送消息:channel.basicPublish()

接收消息:channel.basicConsume()



二、工作模式(一对多)

一个服务者对应多个消费者

在简单模式的基础上多加一个消费者,即创建模块consumer2,所有配置信息与consumer1一致。这里需要注意的是,consumer1和consumer2接收的队列、producer发送的目标队列都应为同一队列。

rabbit mq java整合_rabbitmq

然后启动consumer1和consumer2,再启动producer,可以实时的看到,发送的消息,由两个消费者随机消费。



三、订阅模式

一个服务者对应多个消费者,但是中间由交换机转发。服务者发送消息->交换机->同时发给多个消费者。

准备工作:在测试之前,先确保两个队列在同一个交换机底下(本次模拟交换机为exc1,队列为queue1和queue2)

rabbit mq java整合_java_02

在工作模式的基础上做小修改:

consumer1模块接收队列queue1的消息、consumer2模块接收队列queue2的消息。如下图:

rabbit mq java整合_rabbit mq java整合_03


rabbit mq java整合_rabbit mq java整合_04



注明消息生产者的交换机名称:

rabbit mq java整合_rabbit mq java整合_05

然后启动consumer1和consumer2,再启动producer,可以实时的看到,发送的消息,由两个消费者共同消费,服务者发送的任何消息,所有的消费者都能接收到。



四、路由模式

一个服务者对应多个消费者,中间由交换机转发,交换机有对应的key,指定到对应key的队列。服务者发送消息(指定key)->交换机(根据key转发)->由key发给对应的消费者。

准备工作:在测试之前,先确保两个队列在同一个交换机底下,并且每个队列都有对应的routing key(本次模拟交换机为exc1,队列为queue1和queue2,queue1的key为a,queue2的key为b)

rabbit mq java整合_rabbit mq java整合_06



在订阅模式的基础上做小修改:

producer发送消息时指定对应的key:

//发送消息
public class SendMsg {

    public static void main(String[] args) {
        System.out.println("请输入消息:");
        Scanner input = new Scanner(System.in);
        String msg = input.nextLine();

//        String msg = "hello RabbitMq123!";

        Connection connection = null;

        try {
            //获取连接,相当于JDBC的获取数据库连接
            connection = MQUtil.getConnection();
            /**
             * 相当于JDBC的statement
             * 声明(创建)队列,如果存在就不创建,不存在就创建
             * 参数1 队列名,
             * 参数2 durable: 是否持久化, 默认是存放到内存中,如果rabbitmq重启会丢失,如果想重启之后还存在就要使队列持久化,保存到Erlang自带的Mnesia数据库中,当rabbitmq重启之后会读取该数据库
             * 参数3 exclusive:是否排外的,当当前链接关闭时,是否删除这个队列
             * 参数4 autoDelete:是否自动删除,当最后一个消费者断开连接之后队列是否自动被删除(无论队列中是否有数据)
             * 参数5 arguments: 参数
             * channel.queueDeclare("queue1", false, false, true, null);
             */
            Channel channel = connection.createChannel();
            /**
             * 四个参数:
             * 参数1 : 交换机名称,
             * 参数2 : 目标队列名称,
             * 参数3 : 设置当前这条消息的属性(比如设置超时时间)
             * 参数4 : 消息内容
             */
            if (msg.startsWith("a")){
                channel.basicPublish("exc1","a",null, msg.getBytes());
            } else if (msg.startsWith("b")){
                channel.basicPublish("exc1","b",null, msg.getBytes());
            }

            System.out.println("发送--->" + msg);

            //关闭
            channel.close();
            connection.close();

        } catch (IOException e) {
            e.printStackTrace();
        } catch (TimeoutException e) {
            e.printStackTrace();
        }
    }
}

则消费者只接收到自己key的消息。