Java最全安装RabbitMQ(Centos6)(入门使用教程)(消息丢失的解决方案,mongodb面试题

最后

这份《“java高分面试指南”-25分类227页1000+题50w+字解析》同样可分享给有需要的朋友,感兴趣的伙伴们可挑战一下自我,在不看答案解析的情况,测试测试自己的解题水平,这样也能达到事半功倍的效果!(好东西要大家一起看才香)

image

image

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

import java.io.IOException;

import com.rabbitmq.client.AMQP.BasicProperties;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.DefaultConsumer;

import com.rabbitmq.client.Envelope;

import cn.itcast.rabbitmq.util.ConnectionUtil;

//消费者2

public class Recv2 {

private final static String QUEUE_NAME = “test_work_queue”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

final Channel channel = connection.createChannel();

// 声明队列

channel.queueDeclare(QUEUE_NAME, false, false, false, null);

// 设置每个消费者同时只能处理一条消息

channel.basicQos(1);

// 定义队列的消费者

DefaultConsumer consumer = new DefaultConsumer(channel) {

// 获取消息,并且处理,这个方法类似事件监听,如果有消息的时候,会被自动调用

@Override

public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties,

byte[] body) throws IOException {

// body 即消息体

String msg = new String(body);

System.out.println(" [消费者2] received : " + msg + “!”);

// 手动ACK

channel.basicAck(envelope.getDeliveryTag(), false);

}

};

// 监听队列。

channel.basicConsume(QUEUE_NAME, false, consumer);

}

}

运行测试

同时启动Recv和Recv2

在这里插入图片描述

然后启动Send发送50条信息

在这里插入图片描述

消费者1 和消费者2都在拼命消费

在这里插入图片描述

在这里插入图片描述

4、订阅模型分类

订阅模型示意图:

在这里插入图片描述

前面2个案例中,只有3个角色:

  • P:生产者,也就是要发送消息的程序

  • C:消费者:消息的接受者,会一直等待消息到来。

  • queue:消息队列,图中红色部分。类似一个邮箱,可以缓存消息;生产者向其中投递消息,消费者从其中取出消息。

而在订阅模型中,多了一个exchange角色,而且过程略有变化:

  • P:生产者,也就是要发送消息的程序,但是不再发送到队列中,而是发给X(交换机)

  • C:消费者,消息的接受者,会一直等待消息到来。

  • Queue:消息队列,接收消息、缓存消息。

  • Exchange:交换机,图中的X。一方面,接收生产者发送的消息。另一方面,知道如何处理消息,例如递交给某个特别队列、递交给所有队列、或是将消息丢弃。到底如何操作,取决于Exchange的类型。Exchange有以下3种类型:

  • Fanout:广播,将消息交给所有绑定到交换机的队列

  • Direct:定向,把消息交给符合指定routing key 的队列

  • Topic:通配符,把消息交给符合routing pattern(路由模式) 的队列

Exchange(交换机)只负责转发消息,不具备存储消息的能力,因此如果没有任何队列与Exchange绑定,或者没有符合路由规则的队列,那么消息会丢失!

5、订阅模型-Fanout

Fanout,也称为广播。

流程说明

流程图:

在这里插入图片描述

在广播模式下,消息发送流程是这样的:

  • 1) 可以有多个消费者

  • 2) 每个消费者有自己的queue(队列)

  • 3) 每个队列都要绑定到Exchange(交换机)

  • 4) 生产者发送的消息,只能发送到交换机,交换机来决定要发给哪个队列,生产者无法决定。

  • 5) 交换机把消息发送给绑定过的所有队列

  • 6) 队列的消费者都能拿到消息。实现一条消息被多个消费者消费

在这里插入图片描述

package cn.itcast.rabbitmq.fanout;

import cn.itcast.rabbitmq.util.ConnectionUtil;

import com.rabbitmq.client.BuiltinExchangeType;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

public class Send {

private final static String EXCHANGE_NAME = “fanout_exchange_test”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明exchange(交换机),指定类型为fanout

channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.FANOUT);

// 消息内容

String message = “Hello everyone”;

// 发布消息到Exchange(交换机)

channel.basicPublish(EXCHANGE_NAME, “”, null, message.getBytes());

System.out.println(" [生产者] Sent ‘" + message + "’");

channel.close();

connection.close();

}

}

运行上述代码

在这里插入图片描述

交换机创建成功

在这里插入图片描述

消费者1

在这里插入图片描述

package cn.itcast.rabbitmq.fanout;

import java.io.IOException;

import com.rabbitmq.client.AMQP.BasicProperties;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.DefaultConsumer;

import com.rabbitmq.client.Envelope;

import cn.itcast.rabbitmq.util.ConnectionUtil;

//消费者1

public class Recv {

private final static String QUEUE_NAME = “fanout_exchange_queue_1”;

private final static String EXCHANGE_NAME = “fanout_exchange_test”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明队列

channel.queueDeclare(QUEUE_NAME, false, false, false, null);

// 绑定队列到交换机

channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, “”);

// 定义队列的消费者

DefaultConsumer consumer = new DefaultConsumer(channel) {

// 获取消息,并且处理,这个方法类似事件监听,如果有消息的时候,会被自动调用

@Override

public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties,

byte[] body) throws IOException {

// body 即消息体

String msg = new String(body);

System.out.println(" [消费者1] received : " + msg + “!”);

}

};

// 监听队列,自动返回完成

channel.basicConsume(QUEUE_NAME, true, consumer);

}

}

在这里插入图片描述

package cn.itcast.rabbitmq.fanout;

import java.io.IOException;

import com.rabbitmq.client.AMQP.BasicProperties;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.DefaultConsumer;

import com.rabbitmq.client.Envelope;

import cn.itcast.rabbitmq.util.ConnectionUtil;

// 消费者2

public class Recv2 {

private final static String QUEUE_NAME = “fanout_exchange_queue_2”;

private final static String EXCHANGE_NAME = “fanout_exchange_test”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明队列

channel.queueDeclare(QUEUE_NAME, false, false, false, null);

// 绑定队列到交换机

channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, “”);

// 定义队列的消费者

DefaultConsumer consumer = new DefaultConsumer(channel) {

// 获取消息,并且处理,这个方法类似事件监听,如果有消息的时候,会被自动调用

@Override

public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties,

byte[] body) throws IOException {

// body 即消息体

String msg = new String(body);

System.out.println(" [消费者2] received : " + msg + “!”);

}

};

// 监听队列,手动返回完成

channel.basicConsume(QUEUE_NAME, true, consumer);

}

}

同时运行消费者1和消费者2

在这里插入图片描述

运行Send

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

6、订阅模型-Direct

说明

在Fanout模式中,一条消息,会被所有订阅的队列都消费。但是,在某些场景下,我们希望不同的消息被不同的队列消费。这时就要用到Direct类型的Exchange。

在Direct模型下:

  • 队列与交换机的绑定,不能是任意绑定了,而是要指定一个RoutingKey(路由key)

  • 消息的发送方在 向 Exchange发送消息时,也必须指定消息的 RoutingKey

  • Exchange不再把消息交给每一个绑定的队列,而是根据消息的Routing Key进行判断,只有队列的Routingkey与消息的 Routing key完全一致,才会接收到消息

流程图:

在这里插入图片描述

图解:

  • P:生产者,向Exchange发送消息,发送消息时,会指定一个routing key。

  • X:Exchange(交换机),接收生产者的消息,然后把消息递交给 与routing key完全匹配的队列

  • C1:消费者,其所在队列指定了需要routing key 为 error 的消息

  • C2:消费者,其所在队列指定了需要routing key 为 info、error、warning 的消息

生产者

此处我们模拟商品的增删改,发送消息的RoutingKey分别是:insert、update、delete

在这里插入图片描述

package cn.itcast.rabbitmq.direct;

import cn.itcast.rabbitmq.util.ConnectionUtil;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

/**

  • 生产者,模拟为商品服务

*/

public class Send {

private final static String EXCHANGE_NAME = “direct_exchange_test”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明exchange,指定类型为direct

channel.exchangeDeclare(EXCHANGE_NAME, “direct”);

// 消息内容

String message = “商品删除了, id = 1001”;

// 发送消息,并且指定routing key 为:insert ,代表新增商品

channel.basicPublish(EXCHANGE_NAME, “delete”, null, message.getBytes());

System.out.println(" [商品服务:] Sent ‘" + message + "’");

channel.close();

connection.close();

}

}

运行上述代码创建对应的交换机

在这里插入图片描述

在这里插入图片描述

消费者1

我们此处假设消费者1只接收两种类型的消息:更新商品和删除商品。

在这里插入图片描述

package cn.itcast.rabbitmq.direct;

import java.io.IOException;

import com.rabbitmq.client.AMQP.BasicProperties;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.DefaultConsumer;

import com.rabbitmq.client.Envelope;

import cn.itcast.rabbitmq.util.ConnectionUtil;

/**

  • 消费者1

*/

public class Recv {

private final static String QUEUE_NAME = “direct_exchange_queue_1”;

private final static String EXCHANGE_NAME = “direct_exchange_test”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明队列

channel.queueDeclare(QUEUE_NAME, false, false, false, null);

// 绑定队列到交换机,同时指定需要订阅的routing key。假设此处需要update和delete消息

channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, “update”);

channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, “delete”);

// 定义队列的消费者

DefaultConsumer consumer = new DefaultConsumer(channel) {

// 获取消息,并且处理,这个方法类似事件监听,如果有消息的时候,会被自动调用

@Override

public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties,

byte[] body) throws IOException {

// body 即消息体

String msg = new String(body);

System.out.println(" [消费者1] received : " + msg + “!”);

}

};

// 监听队列,自动ACK

channel.basicConsume(QUEUE_NAME, true, consumer);

}

}

消费者2

我们此处假设消费者2接收所有类型的消息:新增商品,更新商品和删除商品。

在这里插入图片描述

package cn.itcast.rabbitmq.direct;

import java.io.IOException;

import com.rabbitmq.client.AMQP.BasicProperties;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.DefaultConsumer;

import com.rabbitmq.client.Envelope;

import cn.itcast.rabbitmq.util.ConnectionUtil;

/**

  • 消费者2

*/

public class Recv2 {

private final static String QUEUE_NAME = “direct_exchange_queue_2”;

private final static String EXCHANGE_NAME = “direct_exchange_test”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明队列

channel.queueDeclare(QUEUE_NAME, false, false, false, null);

// 绑定队列到交换机,同时指定需要订阅的routing key。订阅 insert、update、delete

channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, “insert”);

channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, “update”);

channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, “delete”);

// 定义队列的消费者

DefaultConsumer consumer = new DefaultConsumer(channel) {

// 获取消息,并且处理,这个方法类似事件监听,如果有消息的时候,会被自动调用

@Override

public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties,

byte[] body) throws IOException {

// body 即消息体

String msg = new String(body);

System.out.println(" [消费者2] received : " + msg + “!”);

}

};

// 监听队列,自动ACK

channel.basicConsume(QUEUE_NAME, true, consumer);

}

}

同时运行

同时运行消费者一和消费者二

在这里插入图片描述

运行Send生产者,模拟为商品服务

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

同时运行修改Send设置为商品新增

在这里插入图片描述

package cn.itcast.rabbitmq.direct;

import cn.itcast.rabbitmq.util.ConnectionUtil;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

/**

  • 生产者,模拟为商品服务

*/

public class Send {

private final static String EXCHANGE_NAME = “direct_exchange_test”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明exchange,指定类型为direct

channel.exchangeDeclare(EXCHANGE_NAME, “direct”);

// 消息内容

String message = “商品新增了, id = 1001”;

// 发送消息,并且指定routing key 为:insert ,代表新增商品

channel.basicPublish(EXCHANGE_NAME, “insert”, null, message.getBytes());

System.out.println(" [商品服务:] Sent ‘" + message + "’");

channel.close();

connection.close();

}

}

运行send

只有Recv2

在这里插入图片描述

Recv没有

在这里插入图片描述

7、订阅模型-Topic

说明

Topic类型的ExchangeDirect相比,都是可以根据RoutingKey把消息路由到不同的队列。只不过Topic类型Exchange可以让队列在绑定Routing key 的时候使用通配符!

Routingkey 一般都是由一个或多个单词组成,多个单词之间以”.”分割,例如: item.insert

通配符规则:

#:匹配一个或多个词

*:匹配不多不少恰好1个词

举例:

audit.#:能够匹配audit.irs.corporate 或者 audit.irs

audit.*:只能匹配audit.irs

图示:

在这里插入图片描述

解释:

  • 红色Queue:绑定的是usa.# ,因此凡是以 usa.开头的routing key 都会被匹配到

  • 黄色Queue:绑定的是#.news ,因此凡是以 .news结尾的 routing key 都会被匹配

生产者

使用topic类型的Exchange,发送消息的routing key有3种: item.isnertitem.updateitem.delete

在这里插入图片描述

package cn.itcast.rabbitmq.topic;

import cn.itcast.rabbitmq.util.ConnectionUtil;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

/**

  • 生产者,模拟为商品服务

*/

public class Send {

private final static String EXCHANGE_NAME = “topic_exchange_test”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明exchange,指定类型为topic

channel.exchangeDeclare(EXCHANGE_NAME, “topic”);

// 消息内容

String message = “新增商品 : id = 1001”;

// 发送消息,并且指定routing key 为:insert ,代表新增商品

channel.basicPublish(EXCHANGE_NAME, “item.insert”, null, message.getBytes());

System.out.println(" [商品服务:] Sent ‘" + message + "’");

channel.close();

connection.close();

}

}

消费者1

我们此处假设消费者1只接收两种类型的消息:更新商品和删除商品

在这里插入图片描述

package cn.itcast.rabbitmq.topic;

import java.io.IOException;

import com.rabbitmq.client.AMQP.BasicProperties;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.DefaultConsumer;

import com.rabbitmq.client.Envelope;

import cn.itcast.rabbitmq.util.ConnectionUtil;

/**

  • 消费者1

*/

public class Recv {

private final static String QUEUE_NAME = “topic_exchange_queue_1”;

private final static String EXCHANGE_NAME = “topic_exchange_test”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明队列

channel.queueDeclare(QUEUE_NAME, false, false, false, null);

// 绑定队列到交换机,同时指定需要订阅的routing key。需要 update、delete

channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, “item.update”);

channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, “item.delete”);

// 定义队列的消费者

DefaultConsumer consumer = new DefaultConsumer(channel) {

// 获取消息,并且处理,这个方法类似事件监听,如果有消息的时候,会被自动调用

@Override

public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties,

byte[] body) throws IOException {

// body 即消息体

String msg = new String(body);

System.out.println(" [消费者1] received : " + msg + “!”);

}

};

// 监听队列,自动ACK

channel.basicConsume(QUEUE_NAME, true, consumer);

}

}

消费者2

我们此处假设消费者2接收所有类型的消息:新增商品,更新商品和删除商品。

在这里插入图片描述

package cn.itcast.rabbitmq.topic;

import java.io.IOException;

import com.rabbitmq.client.AMQP.BasicProperties;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.DefaultConsumer;

import com.rabbitmq.client.Envelope;

import cn.itcast.rabbitmq.util.ConnectionUtil;

/**

  • 消费者2

*/

public class Recv2 {

private final static String QUEUE_NAME = “topic_exchange_queue_2”;

private final static String EXCHANGE_NAME = “topic_exchange_test”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明队列

channel.queueDeclare(QUEUE_NAME, false, false, false, null);

// 绑定队列到交换机,同时指定需要订阅的routing key。订阅 insert、update、delete

channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, “item.*”);

// 定义队列的消费者

DefaultConsumer consumer = new DefaultConsumer(channel) {

// 获取消息,并且处理,这个方法类似事件监听,如果有消息的时候,会被自动调用

@Override

public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties,

byte[] body) throws IOException {

// body 即消息体

String msg = new String(body);

System.out.println(" [消费者2] received : " + msg + “!”);

}

};

// 监听队列,自动ACK

channel.basicConsume(QUEUE_NAME, true, consumer);

}

}

运行测试

运行Send创建交换机

在这里插入图片描述

在这里插入图片描述

运行消费者1和消费者2

在这里插入图片描述

再次运行Send

在这里插入图片描述

消费者2 有消息

在这里插入图片描述

消费者1没有

在这里插入图片描述

修改Send

在这里插入图片描述

package cn.itcast.rabbitmq.topic;

import cn.itcast.rabbitmq.util.ConnectionUtil;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

/**

  • 生产者,模拟为商品服务

*/

public class Send {

private final static String EXCHANGE_NAME = “topic_exchange_test”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明exchange,指定类型为topic

channel.exchangeDeclare(EXCHANGE_NAME, “topic”);

// 消息内容

String message = “删除商品 : id = 1001”;

// 发送消息,并且指定routing key 为:insert ,代表新增商品

channel.basicPublish(EXCHANGE_NAME, “item.delete”, null, message.getBytes());

System.out.println(" [商品服务:] Sent ‘" + message + "’");

channel.close();

connection.close();

}

}

运行Send

在这里插入图片描述

消费者1和消费者2 当中都有消息

在这里插入图片描述

在这里插入图片描述

8、持久化

如何避免消息丢失?

1) 消费者的ACK机制。可以防止消费者丢失消息。

2) 但是,如果在消费者消费之前,MQ就宕机了,消息就没了。

是可以将消息进行持久化呢?

要将消息持久化,前提是:队列、Exchange都持久化

在虚拟机当中重启RabbitMQ

在这里插入图片描述

交换机数据全部消失了

在这里插入图片描述

交换机持久化

在这里插入图片描述

在这里插入图片描述

运行上面代码

package cn.itcast.rabbitmq.topic;

import cn.itcast.rabbitmq.util.ConnectionUtil;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

/**

  • 生产者,模拟为商品服务

*/

public class Send {

private final static String EXCHANGE_NAME = “topic_exchange_test01”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明exchange,指定类型为topic

channel.exchangeDeclare(EXCHANGE_NAME, “topic”,true);

// 消息内容

String message = “删除商品 : id = 1001”;

// 发送消息,并且指定routing key 为:insert ,代表新增商品

channel.basicPublish(EXCHANGE_NAME, “item.delete”, null, message.getBytes());

System.out.println(" [商品服务:] Sent ‘" + message + "’");

channel.close();

connection.close();

}

}

在这里插入图片描述

自己 创建的也有了D(代表持久的)

在这里插入图片描述

队列持久化

在这里插入图片描述

在这里插入图片描述

package cn.itcast.rabbitmq.topic;

import java.io.IOException;

import com.rabbitmq.client.AMQP.BasicProperties;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.DefaultConsumer;

import com.rabbitmq.client.Envelope;

import cn.itcast.rabbitmq.util.ConnectionUtil;

/**

  • 消费者2

*/

public class Recv2 {

private final static String QUEUE_NAME = “topic_exchange_queue_02”;

private final static String EXCHANGE_NAME = “topic_exchange_test01”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明队列

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

// 绑定队列到交换机,同时指定需要订阅的routing key。订阅 insert、update、delete

channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, “item.*”);

// 定义队列的消费者

DefaultConsumer consumer = new DefaultConsumer(channel) {

// 获取消息,并且处理,这个方法类似事件监听,如果有消息的时候,会被自动调用

@Override

public void handleDelivery(String consumerTag, Envelope envelope, BasicProperties properties,

byte[] body) throws IOException {

// body 即消息体

String msg = new String(body);

System.out.println(" [消费者2] received : " + msg + “!”);

}

};

// 监听队列,自动ACK

channel.basicConsume(QUEUE_NAME, true, consumer);

}

}

运行多了一个队列并且也是持久的

在这里插入图片描述

消息持久化

在这里插入图片描述

在这里插入图片描述

package cn.itcast.rabbitmq.topic;

import cn.itcast.rabbitmq.util.ConnectionUtil;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.MessageProperties;

/**

  • 生产者,模拟为商品服务

*/

public class Send {

private final static String EXCHANGE_NAME = “topic_exchange_test01”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

// 声明exchange,指定类型为topic

channel.exchangeDeclare(EXCHANGE_NAME, “topic”, true);

// 消息内容

String message = “新增商品 : id = 1001”;

// 发送消息,并且指定routing key 为:insert ,代表新增商品

channel.basicPublish(EXCHANGE_NAME, “item.insert”, MessageProperties.PERSISTENT_TEXT_PLAIN, message.getBytes());

System.out.println(" [商品服务:] Sent ‘" + message + "’");

channel.close();

connection.close();

}

}

运行

在这里插入图片描述

RabbitMQ挂了或者重启消息也不会消失

生产者确认

//生产者确认

channel.confirmSelect();

channel.waitForConfirms(1000);//失败等待1秒后抛出异常

在这里插入图片描述

package cn.itcast.rabbitmq.topic;

import cn.itcast.rabbitmq.util.ConnectionUtil;

import com.rabbitmq.client.Channel;

import com.rabbitmq.client.Connection;

import com.rabbitmq.client.MessageProperties;

/**

  • 生产者,模拟为商品服务

*/

public class Send {

private final static String EXCHANGE_NAME = “topic_exchange_test01”;

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

// 获取到连接

Connection connection = ConnectionUtil.getConnection();

// 获取通道

Channel channel = connection.createChannel();

//开启生产者确认

channel.confirmSelect();

// 声明exchange,指定类型为topic

channel.exchangeDeclare(EXCHANGE_NAME, “topic”, true);

// 消息内容

String message = “新增商品 : id = 1001”;

// 发送消息,并且指定routing key 为:insert ,代表新增商品

channel.basicPublish(EXCHANGE_NAME, “item.insert”, MessageProperties.PERSISTENT_TEXT_PLAIN, message.getBytes());

System.out.println(" [商品服务:] Sent ‘" + message + "’");

channel.waitForConfirms(1000);//失败等待1秒后抛出异常

channel.close();

connection.close();

}

}

消息丢失的解决方案
  • ack(消费者确认)

  • 持久化

  • 发送消息前,将消息持久化到数据库,并记录消息状态(可靠消息服务)

  • 生产者确认(publisher confirm)

幂等性(同一接口被重复执行,其结果—致)﹔

三、Spring AMQP


1、简介

https://spring.io/projects

Sprin有很多不同的项目,其中就有对AMQP的支持:

在这里插入图片描述

Spring AMQP的页面:http://projects.spring.io/spring-amqp/

在这里插入图片描述

Spring-amqp是对AMQP协议的抽象实现,而spring-rabbit 是对协议的具体实现,也是目前的唯一实现。底层使用的就是RabbitMQ

2、依赖和配置

添加AMQP的启动器:

org.springframework.boot

spring-boot-starter-amqp

在这里插入图片描述

application.yml中添加RabbitMQ地址:

在这里插入图片描述

spring:

rabbitmq:

host: 192.168.206.66

username: leyou

password: 123321

virtual-host: /leyou

3、监听者

在SpringAmqp中,对消息的消费者进行了封装和抽象,一个普通的JavaBean中的普通方法,只要通过简单的注解,就可以成为一个消费者。

在这里插入图片描述

package cn.itcast.rabbitmq.spring;

import org.springframework.amqp.core.ExchangeTypes;

import org.springframework.amqp.rabbit.annotation.Exchange;

import org.springframework.amqp.rabbit.annotation.Queue;

import org.springframework.amqp.rabbit.annotation.QueueBinding;

import org.springframework.amqp.rabbit.annotation.RabbitListener;

import org.springframework.stereotype.Component;

@Component

public class Listener {

@RabbitListener(bindings = @QueueBinding(

value = @Queue(value = “spring.test.queue”, durable = “true”),

exchange = @Exchange(

value = “spring.test.exchange”,

ignoreDeclarationExceptions = “true”,

type = ExchangeTypes.TOPIC

),

key = {“#.#”}))

public void listen(String msg){

System.out.println(“接收到消息:” + msg);

}

}

  • @Componet:类上的注解,注册到Spring容器

  • @RabbitListener:方法上的注解,声明这个方法是一个消费者方法,需要指定下面的属性:

  • bindings:指定绑定关系,可以有多个。值是@QueueBinding的数组。@QueueBinding包含下面属性:

  • value:这个消费者关联的队列。值是@Queue,代表一个队列

  • exchange:队列所绑定的交换机,值是@Exchange类型

  • key:队列和交换机绑定的RoutingKey

类似listen这样的方法在一个类中可以写多个,就代表多个消费者。

4、AmqpTemplate

Spring最擅长的事情就是封装,把他人的框架进行封装和整合。

Spring为AMQP提供了统一的消息处理模板:AmqpTemplate,非常方便的发送消息,其发送方法:

在这里插入图片描述

红框圈起来的是比较常用的3个方法,分别是:

  • 指定交换机、RoutingKey和消息体

  • 指定消息

  • 指定RoutingKey和消息,会向默认的交换机发送消息

5、测试代码

(1)创建队列

在这里插入图片描述

在这里插入图片描述

(2)测试代码(发送消息)

在这里插入图片描述

package cn.itcast.rabbitmq.spring;

import org.junit.Test;

import org.junit.runner.RunWith;

import org.springframework.amqp.core.AmqpTemplate;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.boot.test.context.SpringBootTest;

import org.springframework.test.context.junit4.SpringRunner;

/**

  • @author: HuYi.Zhang

  • @create: 2018-05-23 18:08

**/

@RunWith(SpringRunner.class)

@SpringBootTest(classes = Application.class)

public class MqDemo {

@Autowired

private AmqpTemplate amqpTemplate;

@Test

public void testSend() throws InterruptedException {

String msg = “hello, Spring boot amqp”;

this.amqpTemplate.convertAndSend(“simple_query”, msg);

// 等待10秒后再结束

//Thread.sleep(10000);

}

}

运行

在这里插入图片描述

运行成功

在这里插入图片描述

可以看到

在这里插入图片描述

点击simple_query

在这里插入图片描述

(3)测试代码(接收消息)

在这里插入图片描述

package cn.itcast.rabbitmq.spring;

import org.junit.Test;

import org.springframework.amqp.core.ExchangeTypes;

import org.springframework.amqp.rabbit.annotation.Exchange;

import org.springframework.amqp.rabbit.annotation.Queue;

import org.springframework.amqp.rabbit.annotation.QueueBinding;

import org.springframework.amqp.rabbit.annotation.RabbitListener;

import org.springframework.stereotype.Component;

@Component

public class Listener {

@RabbitListener(queues = “simple_query”)

public void listen(String msg){

System.out.println(“接收到消息:” + msg);

}

}

在设置暂停10秒

在这里插入图片描述

package cn.itcast.rabbitmq.spring;

import org.junit.Test;

import org.junit.runner.RunWith;

import org.springframework.amqp.core.AmqpTemplate;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.boot.test.context.SpringBootTest;

import org.springframework.test.context.junit4.SpringRunner;

/**

**/

@RunWith(SpringRunner.class)

@SpringBootTest(classes = Application.class)

public class MqDemo {

@Autowired

private AmqpTemplate amqpTemplate;

@Test

总结:绘上一张Kakfa架构思维大纲脑图(xmind)

image

其实关于Kafka,能问的问题实在是太多了,扒了几天,最终筛选出44问:基础篇17问、进阶篇15问、高级篇12问,个个直戳痛点,不知道如果你不着急看答案,又能答出几个呢?

若是对Kafka的知识还回忆不起来,不妨先看我手绘的知识总结脑图(xmind不能上传,文章里用的是图片版)进行整体架构的梳理

梳理了知识,刷完了面试,如若你还想进一步的深入学习解读kafka以及源码,那么接下来的这份《手写“kafka”》将会是个不错的选择。

  • Kafka入门

  • 为什么选择Kafka

  • Kafka的安装、管理和配置

  • Kafka的集群

  • 第一个Kafka程序

  • Kafka的生产者

  • Kafka的消费者

  • 深入理解Kafka

  • 可靠的数据传递

  • Spring和Kafka的整合

  • SpringBoot和Kafka的整合

  • Kafka实战之削峰填谷

  • 数据管道和流式处理(了解即可)

image

image

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

FFFFF,t_70,g_se,x_16)

package cn.itcast.rabbitmq.spring;

import org.junit.Test;

import org.springframework.amqp.core.ExchangeTypes;

import org.springframework.amqp.rabbit.annotation.Exchange;

import org.springframework.amqp.rabbit.annotation.Queue;

import org.springframework.amqp.rabbit.annotation.QueueBinding;

import org.springframework.amqp.rabbit.annotation.RabbitListener;

import org.springframework.stereotype.Component;

@Component

public class Listener {

@RabbitListener(queues = “simple_query”)

public void listen(String msg){

System.out.println(“接收到消息:” + msg);

}

}

在设置暂停10秒

在这里插入图片描述

package cn.itcast.rabbitmq.spring;

import org.junit.Test;

import org.junit.runner.RunWith;

import org.springframework.amqp.core.AmqpTemplate;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.boot.test.context.SpringBootTest;

import org.springframework.test.context.junit4.SpringRunner;

/**

**/

@RunWith(SpringRunner.class)

@SpringBootTest(classes = Application.class)

public class MqDemo {

@Autowired

private AmqpTemplate amqpTemplate;

@Test

总结:绘上一张Kakfa架构思维大纲脑图(xmind)

[外链图片转存中…(img-CJyTsRTE-1715345164568)]

其实关于Kafka,能问的问题实在是太多了,扒了几天,最终筛选出44问:基础篇17问、进阶篇15问、高级篇12问,个个直戳痛点,不知道如果你不着急看答案,又能答出几个呢?

若是对Kafka的知识还回忆不起来,不妨先看我手绘的知识总结脑图(xmind不能上传,文章里用的是图片版)进行整体架构的梳理

梳理了知识,刷完了面试,如若你还想进一步的深入学习解读kafka以及源码,那么接下来的这份《手写“kafka”》将会是个不错的选择。

  • Kafka入门

  • 为什么选择Kafka

  • Kafka的安装、管理和配置

  • Kafka的集群

  • 第一个Kafka程序

  • Kafka的生产者

  • Kafka的消费者

  • 深入理解Kafka

  • 可靠的数据传递

  • Spring和Kafka的整合

  • SpringBoot和Kafka的整合

  • Kafka实战之削峰填谷

  • 数据管道和流式处理(了解即可)

[外链图片转存中…(img-fQEvHFxV-1715345164568)]

[外链图片转存中…(img-9km6hob2-1715345164568)]

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

  • 17
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值