SpringBoot消息篇Ⅲ --- 整合RabbitMQ

知识储备: 

关于消息队列的基本概念我已经在上一篇文章介绍过了(传送门),本篇文章主要讲述的是SpringBoot与RabbitMQ的整合以及简单的使用。

一.安装RabbitMQ

1.在linux上使用docker下载RabbitMQ

docker pull registry.docker-cn.com/library/rabbitmq:3-management

2.使用docker启动RabbitMQ

docker run -d -p 5672:5672 -p 15672:15672 --name myrabbitmq d69a5113ceae 

5672端口:客户端与MQ的通信端口

15672端口:管理界面访问web页面的端口

3.访问管理界面

浏览器访问:http://172.16.**.**:15672,默认的管理界面账号密码均为:guest

 

测试RabbitMQ

1). 登录RabbitMQ管理界面,创建交换器(Exchanges)

2). 创建Queues

3). 分别给交换器绑定queues

 

4).在direct交换器中给路由器发送消息

 

5). 队列中接收到的消息

二.环境搭建

1.引入spring-boot-starter-amqp

<dependency>
     <groupId>org.springframework.boot</groupId>
     <artifactId>spring-boot-starter-amqp</artifactId>
</dependency>

2.环境配置

#配置主机地址,默认localhost
spring.rabbitmq.host=172.16.80.34
spring.rabbitmq.username=guest
spring.rabbitmq.password=guest
#默认5672
spring.rabbitmq.port=5672
#默认/
spring.rabbitmq.virtual-host=/

三.RabbitMQ自动配置原理

1.RabbitAutoConfiguration

2.自动配置了连接工厂ConnectionFactory

3.RabbitProperties封装了RabbitMQ的配置

4.RabbitTemplate:给RabbitMQ发送和接收消息的模板

5.AmqpAdmin系统管理组件:创建交换器等

四.RabbitTemplate的简单使用

发送消息:

@Autowired
RabbitTemplate rabbitTemplate;
@Test
public void contextLoads() {

//Message需要自己构造一个,定义消息体内容和消息头
// rabbitTemplate.send(exchange,routingKey,message);
//object默认当成消息体,只需要传入要发送的对象,自动序列化给mq
Map<String,Object> map = new HashMap<>();
map.put("msg","第一次发送消息");
map.put("data",Arrays.asList("<","0.0",">"));
//对象被默认序列化以后发送出去
rabbitTemplate.convertAndSend("exchange.direct","wang.news",map); //使用点对点方式传播
}

此时查看RabbitMQ管理页面的wang.news队列,已经有消息插入进去了,由于RabbitMQ传递的是序列化的对象,所以接收到的值也是序列化过后的值。

接收消息:

 @Test
    public void receive(){
        Object receive = rabbitTemplate.receiveAndConvert("wang.news"); //接收消息。
        System.out.println(receive.getClass());
        System.out.println(receive);
    }

使用该方法获取到消息后队列里的消息就会自动清除。

由于序列化的对象保存起来很不直观,那么该如何解决这个问题呢?

由于RabbitTemplate默认采用的是JDK的MessageConvert,使用默认的JDK序列化规则,所以需要更改MessageConvert,更改为JSON的序列化规则

 

import org.springframework.amqp.support.converter.MessageConverter;//这里要导入amqp包下的MessageConverter
@Configuration
public class MyAMQPConfig {
@Bean
public MessageConverter messageConverter(){ //自动配置里,配置RabbitTemplate的时候会判断是否有自定义的MessageConvert,如果有则采用自定义的
return new Jackson2JsonMessageConverter();
}

}

 

上面演示的是点对点(direct)的交换器(Exchanges),那么广播模式(fanout)的交换器要如何使用的呢?

 @Test
    public void sendMsgs(){
        rabbitTemplate.convertAndSend("exchange.fanout","",new Book("java",2)); //广播模式只需要指定交换器的模式,自动会向该交换器绑定的所有队列发送消息。
    }

发布/订阅(模糊匹配模式)也是一样的,只需要指定交换器,修改对应的routingKey就行了

 /**
     * 发布/订阅(模糊匹配)方式
     */
    @Test
    public void topicSendMsgs(){
        rabbitTemplate.convertAndSend("exchange.topic","*.news",new Book("python",3));
    }

五.监听消息

上面简单演示了使用rabbitTemplate发送和接收消息,实际开发中需要一些监听场景。例如订单系统和库存系统的解耦中,两个系统之间都是通过消息队列来通信的,当某一个人下单之后,将订单信息存放在消息队列中,库存系统要实时的监听消息里面的内容一旦有新的订单进来,库存系统就需要有相关的操作。那么该如何实现监听呢,Spring为了简化开发,引入了一些注解来实现消息队列的监听。

1.在SpringBoot主启动类上加上注解@EnableRabbit,开启RabbitMQ的注解模式

@EnableRabbit//开启基于注解的RabbitMQ模式
@SpringBootApplication
public class Springboot02AmqpApplication {

    public static void main(String[] args) {
        SpringApplication.run(Springboot02AmqpApplication.class, args);
    }

}

2.使用@RabbitListener监听某个队列

@Service
public class BookService {
    @RabbitListener(queues = {"wang.news"}) //监听队列wang.news,只要wang.news收到消息,立刻执行该方法,并清空队列
    public void receive(Book book){
        System.out.println("收到消息"+book);
    }
    @RabbitListener(queues = {"wang"})
    public void receive02(Message message){  //org.springframework.amqp.core.Message;
        System.out.println("消息内容"+message.getBody());
        System.out.println("消息头"+message.getMessageProperties());
    }
}

五.AMQPAdmin的使用

上面代码用到的交换器以及队列都是我们手动在RabbitMQ管理界面添加的,使用AMQPAdmin可以让我们用编码的方式创建这些组件。

1.创建交换器(Exchange)

  @Test
    public void createExchange() {
        DirectExchange directExchange = new DirectExchange("amqpadmin.exchange");
        amqpAdmin.declareExchange(directExchange); //创建一个DirectExchange
        System.out.println("创建完成");
    }

2.创建队列(Queue)

 @Test
    public void createQueue(){
        amqpAdmin.declareQueue(new Queue("amqpadmin.queue",true));
        System.out.println("队列创建完成");
    }

3.创建绑定规则(banding)

 @Test
    public void createBanding(){
        amqpAdmin.declareBinding(new Binding("amqpadmin.queue",Binding.DestinationType.QUEUE,"amqpadmin.exchange","amqpadmin.queue",null));
        System.out.println("绑定完成");
    }

 

posted @ 2019-01-09 17:27  薛定谔病态猫  阅读(764)  评论(0编辑  收藏  举报