rabbitmq教程前五种 来自牧马人老王

Rabbitmq

(一) markdown viewer安装

1.

https://chrome.google.com/webstore/detail/markdown-viewer/ckkdlimhmcjmikdlpkmbgfkaikojcbjk 访问将插件添加到google浏览器

2.

直接在chrome浏览器的网址栏输入chrome://extensions/ 查看已经下载的扩展程序,然后找到Markdown Viewer,点击详细信息,然后将允许访问文件网址勾选。这样以后就可以把想要查看的Markdown文件直接拖进浏览器里就可以查看编译过后好看的排版。

rabbitmq教程前五种 来自牧马人老王_第1张图片
rabbitmq教程前五种 来自牧马人老王_第2张图片

3.

把.md文件在google浏览器上访问,或者在vscode上安装.md插件预览即可

!!4.

推荐使用Typora软件,下载安装即可,方便书写

(二) windows rabbitmq安装

1. http://www.rabbitmq.com

rabbitmq教程前五种 来自牧马人老王_第3张图片
rabbitmq教程前五种 来自牧马人老王_第4张图片
rabbitmq教程前五种 来自牧马人老王_第5张图片
rabbitmq教程前五种 来自牧马人老王_第6张图片
rabbitmq教程前五种 来自牧马人老王_第7张图片

2. 安装完成后,输入以下命令

rabbitmq教程前五种 来自牧马人老王_第8张图片
rabbitmq教程前五种 来自牧马人老王_第9张图片

rabbitmq-plugins enable rabbitmq_management
再在google浏览器上输入 localhost:15672,如果无响应,在自己的C:\Users\(用户名)\AppData\Roaming\RabbitMQ\db路径下的文件全部删除,再重新下载rabbitmq-server,再重复2.即可

rabbitmq教程前五种 来自牧马人老王_第10张图片

3. 看到以下窗口即成功,账户密码皆为guest,此处涉及对rabbitmq的理解,可以理解为mysql数据库。

rabbitmq教程前五种 来自牧马人老王_第11张图片
rabbitmq教程前五种 来自牧马人老王_第12张图片

4. virtual hosts<->mysql中的数据库,一般以/开头,然后对用户进行授权,右边栏中有virtual hosts,进行授权,授权完毕,就可以用自己的账户密码了

rabbitmq教程前五种 来自牧马人老王_第13张图片

(三) java演示

1. 此处是pom.xml的依赖,用maven自动导入,依赖在maven官网可以找到,一定要找groupId和artifactId相同的,且使用非红名测试版的最新版


        <dependency>
            <groupId>com.rabbitmqgroupId>
            <artifactId>amqp-clientartifactId>
            <version>5.7.3version>
        dependency>

        <dependency>
            <groupId>org.slf4jgroupId>
            <artifactId>slf4j-apiartifactId>
            <version>1.7.26version>
        dependency>
        <dependency>
            <groupId>org.slf4jgroupId>
            <artifactId>slf4j-log4j12artifactId>
            <version>1.7.26version>
            <scope>testscope>
        dependency>
        <dependency>
            <groupId>log4jgroupId>
            <artifactId>log4jartifactId>
            <version>1.2.17version>
        dependency>
        <dependency>
            <groupId>junitgroupId>
            <artifactId>junitartifactId>
            <version>4.12version>
            <scope>testscope>
        dependency>

2. 获取MQ连接

写一个工具类util,像tp5里的database.php一样,这样就可以调用了

package com.myapp.demoesi.util;

import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;

import java.io.IOException;
import java.util.concurrent.TimeoutException;
/*包是自动导入的,不用考虑那么多*/

public class ConnectionUtils {

    /*获取MQ的连接,此连接只调用在局部static,并且返回值是一个MQ连接*/
    public static Connection getConnection() throws IOException, TimeoutException {
        /*定义一个连接工厂,相当于mysql中的$conn = mysqli_connect($servername, $username, $password);*/
        ConnectionFactory factory = new ConnectionFactory();
        //设置服务地址$servername
        factory.setHost("127.0.0.1");
        //AMQP 5672(http)
        factory.setPort(5672);

        //vhost(mysql中的数据库)
        factory.setVirtualHost("/vhost_mmr");

        //用户名$username
        factory.setUsername("mmr");
        //密码$password
        factory.setPassword("123456");
        return factory.newConnection();

    }
}

3.1 simple简单队列

rabbitmq教程前五种 来自牧马人老王_第14张图片

P:发送;红色:队列;绿色:消费

新建一个simple包,在里面创建Send和Resv类

3.1.1 Send:

package com.myapp.demoesi.rabbitmq.simple;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Send {
    private static final String QUEUE_NAME = "test_simple_queue";
//私有,静态,不可变
    public static void main(String[] args) throws IOException, TimeoutException {
//获取一个链接,在刚刚的工具类里
        Connection connection = ConnectionUtils.getConnection();
//从连接中获取一个通道
        Channel channel = connection.createChannel();
//创建队列声明
        channel.queueDeclare(QUEUE_NAME, false, false, false, null);
        String msg = "Hello,Simple!";
        //发送在这个队列
        channel.basicPublish("", QUEUE_NAME, null, msg.getBytes());

        System.out.println("--send msg:" + msg);
//信道和连接关了
        channel.close();
        connection.close();
    }
}

运行完毕进行验证,在你的rabbitmq里的Queues,进入你的/vhost_mmr的test_simple_queue里

rabbitmq教程前五种 来自牧马人老王_第15张图片

观察是否可以接收到发送的消息,如果可以Send就成功啦!

rabbitmq教程前五种 来自牧马人老王_第16张图片

3.1.2 Resv:

package com.myapp.demoesi.rabbitmq.simple;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Resv {
    private static final String QUEUE_NAME = "test_simple_queue";

    public static void main(String[] args) throws IOException, TimeoutException {
        //获取链接
        Connection connection = ConnectionUtils.getConnection();
//创建频道
        Channel channel = connection.createChannel();
        //队列声明,保险起见
        channel.queueDeclare(QUEUE_NAME, false, false, false, null);

        DefaultConsumer consumer = new DefaultConsumer(channel) {
            //获取到达的消息
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);
                String msg = new String(body, "UTF-8");
                System.out.println("new api resv:  " + msg);
            }
        };
//监听队列 android
        channel.basicConsume(QUEUE_NAME, true, consumer);
    }


    
 /*下面是老方法,此处供学习参考,redis也差不多这样,这个网址聊天这个挺不错的https://www.imooc.com/learn/758*/
    public void oldresv() {
        /*    //获取链接
        Connection connection = ConnectionUtils.getConnection();
//创建频道
        Channel channel = connection.createChannel();
//定义队列的消费者
QueueingConsumer consumer = new QueueingConsumer(channel);
//监听队列
channel.basicConsume(QUEUE_NAME,true,consumer);
while(true){
    Delivery delivery = consumer.nextDelivery();
String msgString = new String(delivery.getBody());
    System.out.println("[resv]msg:"+msgString);
}*///老方法1.3
    }
}

先开Resv,再执行Send,就可以看到我们发送的Hello simple!了。

3.2.1 work工作队列(轮训分发)

rabbitmq教程前五种 来自牧马人老王_第17张图片

3.2.1.1 Send: 在队列里攒了50个消息

package com.myapp.demoesi.rabbitmq.work;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Send {
/*
*p---queue
 */
private static final String QUEUE_NAME = "test_work_queue";
    public static void main(String[] args) throws IOException, TimeoutException, InterruptedException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();
        //声明队列
        channel.queueDeclare(QUEUE_NAME,false,false,false,null);
        for (int i = 0;i<50;i++){
            String msg = "Hello,"+i;
            System.out.println("[work] send: "+msg);
            channel.basicPublish("",QUEUE_NAME,null,msg.getBytes());

            Thread.sleep(i*20);
        }
        channel.close();
        connection.close();
    }
}

3.2.1.2 Resv1:消费者1

package com.myapp.demoesi.rabbitmq.work;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;

import java.util.concurrent.TimeoutException;

public class Resv1 {
    private static final String QUEUE_NAME = "test_work_queue";

    public static void main(String[] args) throws IOException, TimeoutException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();
        //S声明队列
        channel.queueDeclare(QUEUE_NAME, false, false, false, null);

        //定义一个消费者
        Consumer consumer = new DefaultConsumer(channel) {
            //消息到达触发这个方法
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);

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

                System.out.println("[resv1] msg:" + msg);
                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {
                    System.out.println("[1] done");

                }

            }
        };
        boolean autoAck = true;
        channel.basicConsume(QUEUE_NAME,autoAck,consumer);
    }
}

3.2.2.3 Resv2:消费者2

package com.myapp.demoesi.rabbitmq.work;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Resv2 {
    private static final String QUEUE_NAME = "test_work_queue";

    public static void main(String[] args) throws IOException, TimeoutException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();
        //S声明队列
        channel.queueDeclare(QUEUE_NAME, false, false, false, null);

        //定义一个消费者
        Consumer consumer = new DefaultConsumer(channel) {
            //消息到达触发这个方法
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);

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

                System.out.println("[resv1] msg:" + msg);
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {
                    System.out.println("[2] done");

                }

            }
        };
        boolean autoAck = true;
        channel.basicConsume(QUEUE_NAME,autoAck,consumer);
    }
}

3.2.2 workfair工作队列(公平分发)

rabbitmq教程前五种 来自牧马人老王_第18张图片

3.2.2.1 Send: 在队列里攒了50个消息

package com.myapp.demoesi.rabbitmq.workfair;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.MessageProperties;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Send {
    /*
     *p---queue
     */
    private static final String QUEUE_NAME = "test_workfair_queue";

    public static void main(String[] args) throws IOException, TimeoutException, InterruptedException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();
        //声明队列
        boolean durable = true;
        channel.queueDeclare(QUEUE_NAME, durable, false, false, null);/*消息持久化
        每个消费者发送确认消息之前,消息队列不发送下一个消息到消费者,一次只处理一个消息
durable持久化,但在声明队列后不能更改参数,因为队列已经存在;但消息不一定持久化*/
        int prefetchCount = 1;
        channel.basicQos(prefetchCount);//限制发送给同一个消费者不得超过一个消息


        for (int i = 0; i < 50; i++) {
            String msg = "Hello," + i;
            System.out.println("[work] send: " + msg);
            channel.basicPublish("", QUEUE_NAME, MessageProperties.PERSISTENT_TEXT_PLAIN, msg.getBytes());

            Thread.sleep(i * 5);
        }
        channel.close();
        connection.close();
    }
}

3.2.2.2 Resv1:消费者1

package com.myapp.demoesi.rabbitmq.workfair;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Resv1 {
    private static final String QUEUE_NAME = "test_workfair_queue";

    public static void main(String[] args) throws IOException, TimeoutException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();
        //S声明队列
        boolean durable = true;
        channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化

        channel.basicQos(1);


        //定义一个消费者

        Consumer consumer = new DefaultConsumer(channel) {
            int ing = 1;

            //消息到达触发这个方法
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);

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

                System.out.println("[resv1] msg:" + msg);

                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {

                    System.out.println("[" + ing + "] done");
                    ing++;
                    channel.basicAck(envelope.getDeliveryTag(), false);

                }

            }

        };

        boolean autoAck = false;//自动应答
        channel.basicConsume(QUEUE_NAME, autoAck, consumer);
    }
}

3.2.2.3 Resv2:消费者2

package com.myapp.demoesi.rabbitmq.workfair;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Resv2 {
    private static final String QUEUE_NAME = "test_workfair_queue";

    public static void main(String[] args) throws IOException, TimeoutException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();
        //S声明队列
        boolean durable = true;
        channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化
        channel.basicQos(1);

        //定义一个消费者
        Consumer consumer = new DefaultConsumer(channel) {
            int ing = 1;

            //消息到达触发这个方法
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);

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

                System.out.println("[resv1] msg:" + msg);
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {

                    System.out.println("[" + ing + "] done");
                    ing++;
                    //手动回执一个消息
                    channel.basicAck(envelope.getDeliveryTag(), false);
                }

            }
        };
        boolean autoAck = false;
        channel.basicConsume(QUEUE_NAME, autoAck, consumer);//自动应答
    }
}

3.3 publish_subscribe订阅模式 fanout(不处理路由键)

rabbitmq教程前五种 来自牧马人老王_第19张图片

3.3.1 Send: 发送到交换机

package com.myapp.demoesi.rabbitmq.ps;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.MessageProperties;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Send {
    private static final String EXCHANGE_NAME="test_exchange_fanout";
    public static void main(String[] args) throws IOException, TimeoutException {
        Connection connection = ConnectionUtils.getConnection();
        Channel channel=connection.createChannel();

        //声明交换机
        channel.exchangeDeclare(EXCHANGE_NAME,"fanout");/*分发
        交换机没有存储消息的功能,队列才有
        一方面接收生产者的消息,另一方面是向对列推送消息
        匿名转发:""
        fanout(不处理路由键)
        direct(处理路由键)
        */
        //发送消息
        String msg="Hello ps !";
        channel.basicPublish(EXCHANGE_NAME,"" , MessageProperties.PERSISTENT_TEXT_PLAIN, msg.getBytes());
        System.out.println("Send: "+msg);


        channel.close();
        connection.close();
    }
}

3.3.2 Resv1: 交换机绑定队列

package com.myapp.demoesi.rabbitmq.ps;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Resv1 {
    private static final String QUEUE_NAME = "test_queue_fanout_email";
    private static final String EXCHANGE_NAME = "test_exchange_fanout";

    public static void main(String[] args) throws IOException, TimeoutException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();
        //S声明队列
        boolean durable = true;
        channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化
        //绑定队列到交换机转发器
        channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "");

        channel.basicQos(1);


        //定义一个消费者
        System.out.println("1 666");
        Consumer consumer = new DefaultConsumer(channel) {
            int ing = 1;

            //消息到达触发这个方法
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);

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

                System.out.println("[resv1] msg:" + msg);

                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {

                    System.out.println("[" + ing + "] done");
                    ing++;
                    channel.basicAck(envelope.getDeliveryTag(), false);

                }

            }

        };

        boolean autoAck = false;//自动应答
        channel.basicConsume(QUEUE_NAME, autoAck, consumer);

    }
}

3.3.3 Resv2: 交换机绑定队列

package com.myapp.demoesi.rabbitmq.ps;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Resv2 {
    private static final String QUEUE_NAME = "test_queue_fanout_sms";
    private static final String EXCHANGE_NAME = "test_exchange_fanout";

    public static void main(String[] args) throws IOException, TimeoutException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();
        //S声明队列
        boolean durable = true;
        channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化
        //绑定队列到交换机转发器
        channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "");

        channel.basicQos(1);


        //定义一个消费者
        System.out.println("2 666");
        Consumer consumer = new DefaultConsumer(channel) {
            int ing = 1;

            //消息到达触发这个方法
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);

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

                System.out.println("[resv2] msg:" + msg);

                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {

                    System.out.println("[" + ing + "] done");
                    ing++;
                    channel.basicAck(envelope.getDeliveryTag(), false);

                }

            }

        };

        boolean autoAck = false;//自动应答
        channel.basicConsume(QUEUE_NAME, autoAck, consumer);

    }
}

3.4 路由模式 direct(处理路由键)

rabbitmq教程前五种 来自牧马人老王_第20张图片

3.4.1 Send: 发送携带路由键的消息给交换机

package com.myapp.demoesi.rabbitmq.routing;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.MessageProperties;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Send {
    private static final String EXCHANGE_NAME="test_exchange_direct";
    public static void main(String[] args) throws IOException, TimeoutException {
        Connection connection = ConnectionUtils.getConnection();
        Channel channel = connection.createChannel();
         //exchange
        channel.exchangeDeclare(EXCHANGE_NAME,"direct");
        String routingKey = "info";
        String msg = "Hello, "+routingKey+"! ";
        channel.basicPublish(EXCHANGE_NAME, msg, MessageProperties.PERSISTENT_TEXT_PLAIN, routingKey.getBytes());
        System.out.println("Send: "+msg);

        channel.close();
        connection.close();
    }
}

3.4.2 Resv1: 绑定"error"路由键和队列"test_queue_direct_1"到交换机上

package com.myapp.demoesi.rabbitmq.routing;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Resv1 {
    private static final String EXCHANGE_NAME = "test_exchange_direct";
    private static final String QUEUE_NAME = "test_queue_direct_1";
    public static void main(String[] args) throws IOException, TimeoutException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();

        boolean durable = true;
        channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化

        channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"error");
        channel.basicQos(1);
        Consumer consumer = new DefaultConsumer(channel) {
            int ing = 1;

            //消息到达触发这个方法
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);

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

                System.out.println("[resv1] msg:" + msg);

                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {

                    System.out.println("[" + ing + "] done");
                    ing++;
                    channel.basicAck(envelope.getDeliveryTag(), false);

                }

            }

        };

        boolean autoAck = false;//自动应答
        channel.basicConsume(QUEUE_NAME, autoAck, consumer);
    }
}

3.4.3 Resv2: 绑定"warning"“info”"error"路由键和队列"test_queue_direct_2"到交换机上

package com.myapp.demoesi.rabbitmq.routing;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Resv2 {
    private static final String EXCHANGE_NAME = "test_exchange_direct";
    private static final String QUEUE_NAME = "test_queue_direct_2";
    public static void main(String[] args) throws IOException, TimeoutException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();

        boolean durable = true;
        channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化

        channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"error");
        channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"info");
        channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"warning");
        channel.basicQos(1);
        Consumer consumer = new DefaultConsumer(channel) {
            int ing = 1;

            //消息到达触发这个方法
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);

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

                System.out.println("[resv2] msg:" + msg);

                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {

                    System.out.println("[" + ing + "] done");
                    ing++;
                    channel.basicAck(envelope.getDeliveryTag(), false);

                }

            }

        };

        boolean autoAck = false;//自动应答
        channel.basicConsume(QUEUE_NAME, autoAck, consumer);
    }
}

3.5 topic主题模式 (通配符处理*、#)

rabbitmq教程前五种 来自牧马人老王_第21张图片

3.5.1 Send: routingKey是"goods.delete",可以让goods.#和#.delete绑定的队列接收到

package com.myapp.demoesi.rabbitmq.topic;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.MessageProperties;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Send {
    private static final String EXCHANGE_NAME = "test_exchange_topic";

    public static void main(String[] args) throws IOException, TimeoutException {
        Connection connection = ConnectionUtils.getConnection();
        Channel channel = connection.createChannel();
        //exchange
        channel.exchangeDeclare(EXCHANGE_NAME, "topic");
        String routingKey = "goods.delete";
        String msg = "商品.... " + routingKey + "! ";
        channel.basicPublish(EXCHANGE_NAME, routingKey, MessageProperties.PERSISTENT_TEXT_PLAIN, routingKey.getBytes());
        System.out.println("Send: " + msg);

        channel.close();
        connection.close();
    }
}

3.5.2 Resv1: 绑定goods.add的"test_queue_topic_1"接收goods.add消息

package com.myapp.demoesi.rabbitmq.topic;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Resv1 {
    private static final String EXCHANGE_NAME = "test_exchange_topic";
    private static final String QUEUE_NAME = "test_queue_topic_1";
    public static void main(String[] args) throws IOException, TimeoutException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();

        boolean durable = true;
        channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化

        channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"goods.add");

        channel.basicQos(1);
        Consumer consumer = new DefaultConsumer(channel) {
            int ing = 1;

            //消息到达触发这个方法
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);

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

                System.out.println("[resv1] msg:" + msg);

                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {

                    System.out.println("[" + ing + "] done");
                    ing++;
                    channel.basicAck(envelope.getDeliveryTag(), false);

                }

            }

        };

        boolean autoAck = false;//自动应答
        channel.basicConsume(QUEUE_NAME, autoAck, consumer);
    }
}

3.5.3 Resv2: 绑定goods.#的"test_queue_topic_2"接收goods.#的消息

package com.myapp.demoesi.rabbitmq.topic;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Resv2 {
    private static final String EXCHANGE_NAME = "test_exchange_topic";
    private static final String QUEUE_NAME = "test_queue_topic_2";
    public static void main(String[] args) throws IOException, TimeoutException {
        //获取连接
        Connection connection = ConnectionUtils.getConnection();
        //获取channel
        Channel channel = connection.createChannel();

        boolean durable = true;
        channel.queueDeclare(QUEUE_NAME, durable, false, false, null);//消息持久化

        channel.queueBind(QUEUE_NAME,EXCHANGE_NAME,"goods.#");
        channel.basicQos(1);
        Consumer consumer = new DefaultConsumer(channel) {
            int ing = 1;

            //消息到达触发这个方法
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);

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

                System.out.println("[resv2] msg:" + msg);

                try {
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {

                    System.out.println("[" + ing + "] done");
                    ing++;
                    channel.basicAck(envelope.getDeliveryTag(), false);

                }

            }

        };

        boolean autoAck = false;//自动应答
        channel.basicConsume(QUEUE_NAME, autoAck, consumer);
    }
}

(四) rabbitmq的消息确认机制(事务+confirm)

3.1 在rabbitmq中 我们可以通过持久化数据 解决rabbitmq服务器异常 的数据丢失问题

问题:生产者将消息发送出去之后,消息到底有没有到达 rabbitmq 服务器,默认的情况是不知道的;

两种方式:

​ AMQP实现了事务机制,

​ Confirm模式

3.2 事务机制

txSelect txCommit txRollback

txSelect: 用户将当前channel设置成transation模式

txCommit: 用于提交事务

txRollback: 回滚事务

3.2.1 TxSend:发送translation模式的事务

package com.myapp.demoesi.rabbitmq.tx;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class TxSend {
    private static final String QUEUE_NAME = "test_queue_tx";
    public static void main(String[] args) throws IOException, TimeoutException {
      Connection connection =  ConnectionUtils.getConnection();
       Channel channel = connection.createChannel();
       channel.queueDeclare(QUEUE_NAME,false,false,false,null);
       String msg="Hello tx massage! ";

try {
    channel.txSelect();
    channel.basicPublish("",QUEUE_NAME,null,msg.getBytes());
    //int xx = 1/0;

    channel.txCommit();
}catch (Exception e){
    channel.txRollback();
    System.out.println("  send message rollback");
}

channel.close();
connection.close();


    }
}

3.2.2 TxResv:正常接收

package com.myapp.demoesi.rabbitmq.tx;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class txResv {
    private static final String QUEUE_NAME="test_queue_tx";
    public static void main(String[] args) throws IOException, TimeoutException {
        Connection connection = ConnectionUtils.getConnection();
        Channel channel = connection.createChannel();
channel.queueDeclare(QUEUE_NAME,false,false,false,null);

channel.basicConsume(QUEUE_NAME,true,
        new DefaultConsumer(channel){
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);
                System.out.println("resv[tx]"+new String(body,"utf-8"));

            }
        });

    }

}

此种模式还是很耗时的,采用这种方式,降低了rabbitmq的吞吐量

3.3 Confirm模式

Confirm模式最大的好处在于他是异步

Nack消息

开启confirm模式

channel.confirmSelect()

编程模式:

1.普通 发一条 waitForConfirms()

2.批量的 发一批 waitForConfirms()

3.异步confirm模式:提供一个回调(我并不会)

3.3.1 Confirm单条: Send1

package com.myapp.demoesi.rabbitmq.confirm;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

/*普通模式*/
public class Send1 {
    private static final String QUEUE_NAME = "test_queue_confirm1";
    public static void main(String[] args) throws IOException, TimeoutException, InterruptedException {
        Connection connection =  ConnectionUtils.getConnection();
        Channel channel = connection.createChannel();
        channel.queueDeclare(QUEUE_NAME,false,false,false,null);

//生产者调用confirmSelect将channel设置为confirm模式  注意
        channel.confirmSelect();
        String msg="Hello confirm1 massage! ";


            channel.basicPublish("",QUEUE_NAME,null,msg.getBytes());
            //int xx = 1/0;
if (!channel.waitForConfirms()){
    System.out.println("massage send failed");
}else{
    System.out.println("massage send ok");
}


        channel.close();
        connection.close();
    }
}

3.3.2 Confirm批量: Send2

package com.myapp.demoesi.rabbitmq.confirm;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

/*普通模式*/
public class Send2 {
    private static final String QUEUE_NAME = "test_queue_confirm1";

    public static void main(String[] args) throws IOException, TimeoutException, InterruptedException {
        Connection connection = ConnectionUtils.getConnection();
        Channel channel = connection.createChannel();
        channel.queueDeclare(QUEUE_NAME, false, false, false, null);

//生产者调用confirmSelect将channel设置为confirm模式  注意
        channel.confirmSelect();
        String msg = "Hello confirm2 massage! ";
        for (int i = 0; i < 10; i++) {
            channel.basicPublish("", QUEUE_NAME, null, msg.getBytes());
        }


        //int xx = 1/0;
        if (!channel.waitForConfirms()) {
            System.out.println("massage send failed");
        } else {
            System.out.println("massage send ok");
        }


        channel.close();
        connection.close();
    }
}

3.3.3 Resv:

package com.myapp.demoesi.rabbitmq.confirm;

import com.myapp.demoesi.util.ConnectionUtils;
import com.rabbitmq.client.*;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Resv {
    private static final String QUEUE_NAME="test_queue_confirm1";
    public static void main(String[] args) throws IOException, TimeoutException {
        Connection connection = ConnectionUtils.getConnection();
        Channel channel = connection.createChannel();
channel.queueDeclare(QUEUE_NAME,false,false,false,null);

channel.basicConsume(QUEUE_NAME,true,
        new DefaultConsumer(channel){
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                super.handleDelivery(consumerTag, envelope, properties, body);
                System.out.println("resv[confirm]"+new String(body,"utf-8"));

            }
        });

    }

}

很简单吧?
你学会了吗?

你可能感兴趣的:(消息传递)