JavaWeb_SpringCloud微服务_Day4-MQ, RabbitMQ, SpringAMQP

MQ

同步通讯

  • 优点:
    • 时效性强, 可以立即得到结果
  • 缺点:
    • 耦合度高
    • 性能和吞吐能力下降
    • 有额外的资源消耗
    • 有级联失败问题

异步通讯

  • 优点:
    • 耦合度低
    • 吞吐量提升
    • 故障隔离
    • 流量削峰
  • 缺点:
    • 依赖于Broker的可靠性, 安全性, 吞吐能力
    • 架构复杂, 业务没有明显的流程线, 不好追踪管理

mq常见技术

RabbitMqActiveMQRocketMQKafka
公司/社区RabbitApache阿里Apache
开发语言ErlangjavajavaScala&java
协议支持AMQP, XMPP, SMTP, STOMPOpenWire, STOMP, REST, XMPP, AMQP自定义协议自定义协议
可用性一般
单机吞吐量一般非常高
消息延迟微秒级毫秒级毫秒级毫秒以内
消息可靠性一般一般

RabbitMQ

下载安装

  • docker下载
docker pull rabbitmq:3-management
  • docker运行
docker run \
 -e RABBITMQ_DEFAULT_USER=itcast \
 -e RABBITMQ_DEFAULT_PASS=123321 \
 --name mq \
 --hostname mq1 \
 -p 15672:15672 \
 -p 5672:5672 \
 -d \
 rabbitmq:3-management

介绍

  • channel: 操作MQ的工具
  • exchange: 路由消息到队列中
  • queue: 缓存消息
  • virtual host: 虚拟主机, 是对queue, exchange等资源的逻辑分组

SimpleQueue模型

  • publisher
public void testSendMessage() throws IOException, TimeoutException {
    // 1.建立连接
    ConnectionFactory factory = new ConnectionFactory();
    // 1.1.设置连接参数,分别是:主机名、端口号、vhost、用户名、密码
    factory.setHost("192.168.174.133");
    factory.setPort(5672);
    factory.setVirtualHost("/");
    factory.setUsername("itcast");
    factory.setPassword("123321");
    // 1.2.建立连接
    Connection connection = factory.newConnection();

    // 2.创建通道Channel
    Channel channel = connection.createChannel();

    // 3.创建队列
    String queueName = "simple.queue";
    channel.queueDeclare(queueName, false, false, false, null);

    // 4.发送消息
    String message = "hello, rabbitmq!";
    channel.basicPublish("", queueName, null, message.getBytes());
    System.out.println("发送消息成功:【" + message + "】");

    // 5.关闭通道和连接
    channel.close();
    connection.close();
}
  • consumer
public static void main(String[] args) throws IOException, TimeoutException {
    // 1.建立连接
    ConnectionFactory factory = new ConnectionFactory();
    // 1.1.设置连接参数,分别是:主机名、端口号、vhost、用户名、密码
    factory.setHost("192.168.174.133");
    factory.setPort(5672);
    factory.setVirtualHost("/");
    factory.setUsername("itcast");
    factory.setPassword("123321");
    // 1.2.建立连接
    Connection connection = factory.newConnection();

    // 2.创建通道Channel
    Channel channel = connection.createChannel();

    // 3.创建队列
    String queueName = "simple.queue";
    channel.queueDeclare(queueName, false, false, false, null);

    // 4.订阅消息
    channel.basicConsume(queueName, true, new DefaultConsumer(channel){
        @Override
        public void handleDelivery(String consumerTag, Envelope envelope,
                                    AMQP.BasicProperties properties, byte[] body) throws IOException {
            // 5.处理消息
            String message = new String(body);
            System.out.println("接收到消息:【" + message + "】");
        }
    });
    System.out.println("等待接收消息。。。。");
}

SpringAMQP

介绍

  • AMQP: 应用间消息通信的一种协议, 与语言和平台无关.
  • 依赖:
    <!--AMQP依赖,包含RabbitMQ-->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-amqp</artifactId>
    </dependency>
    

SimpleQueue模型

  • 配置
    spring:
    rabbitmq:
        host: 192.168.174.133 # 主机名S
        port: 5672 # 端口
        virtual-host: / # 虚拟主机
        username: itcast # 用户名
        password: 123321 # 密码
    
  • publisher
    @SpringBootTest
    @RunWith(SpringRunner.class)
    public class SpringAmqpTest {
    
        @Autowired
        private RabbitTemplate rabbitTemplate;
    
        @Test
        public void testSimpleQueue()
        {
            String queueName = "simple.queue";
            String message = "hello, spring amqp!";
            rabbitTemplate.convertAndSend(queueName, message);
        }
    }
    
  • comsumer
    @Component
    public class SpringRabbitListener {
        @RabbitListener(queues = "simple.queue")
        public void listenSimpleQueueMessage(String msg)
        {
            System.out.println("spring 消费者接收到消息: ["+msg+"]");
        }
    }
    

WorkQueue模型

  • publisher
    @Test
    public void testWorkQueue() throws InterruptedException {
        String queueName = "simple.queue";
        String message = "hello, spring amqp!";
        for (int i = 1; i <= 50; i++) {
            rabbitTemplate.convertAndSend(queueName, message);
            Thread.sleep(20);
        }
    }
    
  • consumer
    @RabbitListener(queues = "simple.queue")
    public void listenWorkQueue1(String msg) throws InterruptedException {
        System.out.println("消费者1接收到消息: ["+msg+"]"+ LocalTime.now());
        Thread.sleep(20);
    }
    
    @RabbitListener(queues = "simple.queue")
    public void listenWorkQueue2(String msg) throws InterruptedException {
        System.err.println("消费者2接收到消息: ["+msg+"]"+ LocalTime.now());
        Thread.sleep(200);
    }
    
  • 配置
    spring:
    rabbitmq:
        listener:
        simple:
            prefetch: 1 # 每次只能获取下一消息, 处理完成才能获取下一个消息
    
  • work模型总结:
    • 多个消费者绑定到一个队列, 同一条消息只会被一个消费者处理
    • consumer会预取消息, 会导致性能差的consumer堆积消息, 可以通过设置prefetch来控制消费者预取的消息数量.

发布订阅模型

介绍

  • 发布订阅模式与之前案例的区别就是允许将同一消息发送给多个消费者, 实现方式就是加入了exchange(交换机).
  • 常见exchange类型:
    • Fanout: 广播
    • Direct: 路由
    • Topic: 话题

FanoutExchange

  • config
    @Configuration
    public class FanoutConfig {
    
        @Bean
        public FanoutExchange fanoutExchange()
        {
            return new FanoutExchange("itcast.fanout");
        }
    
        @Bean
        public Queue fanoutQueue1()
        {
            return new Queue("fanout.queue1");
        }
    
        @Bean
        public Queue fanoutQueue2()
        {
            return new Queue("fanout.queue2");
        }
    
        @Bean
        public Binding fanoutBinding1(Queue fanoutQueue1, FanoutExchange fanoutExchange)
        {
            return BindingBuilder.bind(fanoutQueue1).to(fanoutExchange);
        }
        @Bean
        public Binding fanoutBinding2(Queue fanoutQueue2, FanoutExchange fanoutExchange)
        {
            return BindingBuilder.bind(fanoutQueue2).to(fanoutExchange);
        }
    }
    
  • consumer
    @RabbitListener(queues = "fanout.queue1")
    public void listenFanoutQueue1(String msg) throws InterruptedException {
        System.out.println("消费者1接收到消息: ["+msg+"]"+ LocalTime.now());
        Thread.sleep(20);
    }
    @RabbitListener(queues = "fanout.queue2")
    public void listenFanoutQueue2(String msg) throws InterruptedException {
        System.err.println("消费者2接收到消息: ["+msg+"]"+ LocalTime.now());
        Thread.sleep(200);
    }
    
  • publisher
    @Test
    public void testSendFanoutExchange()
    {
        // 交换机名称
        String exchangeName = "itcast.fanout";
        // 消息
        String message = "hello, every one!";
        // 发送消息
        rabbitTemplate.convertAndSend(exchangeName, "", message);
    }
    
  • 总结:
    • 交换机的作用
      • 接收publisher发送的消息
      • 将消息按照规则路由到与之绑定的队列
      • 不能缓存消息, 路由失败, 消息丢失
      • FanoutExchange的会将消息路由到每个绑定的队列
    • 声明队列的Bean: Queue
    • 声明交换机的Bean: FanoutExchange
    • 声明绑定关系的Bean: Binding

Direct Exchange

  • 介绍
    • 每一个Queue都与Exchange设置一个BindingKey
    • 发布者发送消息时, 指定消息的RoutingKey
    • Exchange将消息路由到BindingKey与消息RoutingKey一致的队列
  • consumer
    @RabbitListener(bindings = @QueueBinding(
            value = @Queue(name = "direct.queue1"),
            exchange = @Exchange(name = "itcast.direct", type = ExchangeTypes.DIRECT),
            key = {"red", "blue"}
    ))
    public void listenDirectQueue1(String msg) throws InterruptedException {
        System.out.println("消费者1接收到消息: ["+msg+"]");
    }
    @RabbitListener(bindings = @QueueBinding(
            value = @Queue(name = "direct.queue2"),
            exchange = @Exchange(name = "itcast.direct", type = ExchangeTypes.DIRECT),
            key = {"red", "yellow"}
    ))
    public void listenDirectQueue2(String msg) throws InterruptedException {
        System.err.println("消费者2接收到消息: ["+msg+"]");
    }
    
  • publisher
    @Test
    public void testSendDirectExchange()
    {
        // 交换机名称
        String exchangeName = "itcast.direct";
        // 消息
        String message = "hello, every blue!";
        // 发送消息
        rabbitTemplate.convertAndSend(exchangeName, "blue", message);
    }
    

Topic Exchange

  • consumer
    @RabbitListener(bindings = @QueueBinding(
            value = @Queue(name = "topic.queue1"),
            exchange = @Exchange(name = "itcast.topic", type = ExchangeTypes.TOPIC),
            key = "china.#"
    ))
    public void listenTopicQueue1(String msg) throws InterruptedException {
        System.out.println("消费者1接收到消息: ["+msg+"]");
    }
    @RabbitListener(bindings = @QueueBinding(
            value = @Queue(name = "topic.queue2"),
            exchange = @Exchange(name = "itcast.topic", type = ExchangeTypes.TOPIC),
            key = "#.news"
    ))
    public void listenTopicQueue2(String msg) throws InterruptedException {
        System.err.println("消费者2接收到消息: ["+msg+"]");
    }
    
  • publisher
    @Test
    public void testSendTopicExchange()
    {
        // 交换机名称
        String exchangeName = "itcast.topic";
        // 消息
        String message = "hello, china.news";
        // 发送消息
        rabbitTemplate.convertAndSend(exchangeName, "china.news", message);
    }
    
  • 总结
    • TopicExchange与DirectExchange类似, 区别在于routingKey必须是多个单词的列表, 并且以.分割.

消息转换器

  • 依赖
    <dependency>
        <groupId>com.fasterxml.jackson.core</groupId>
        <artifactId>jackson-databind</artifactId>
    </dependency>
    
  • MessageConverter
    @Bean
    public MessageConverter jsonMessageConverter()
    {
        return new Jackson2JsonMessageConverter();
    }
    
  • publisher
    @Test
    public void testSendObjectQueue()
    {
        Map<String, Object> msg = new HashMap<>();
        msg.put("name", "柳岩");
        msg.put("age", 21);
        rabbitTemplate.convertAndSend("object.queue", msg);
    }
    
  • consumer
    @RabbitListener(queues = "object.queue")
    public void listenObjectQueue(Map<String, Object> msg)
    {
        System.out.println("接收到object.queue的消息: "+msg);
    }
    

来源

黑马程序员. SpringCloud微服务

### 回答1: javaweb_cloud_elevue_pro微服务框架旗舰版源码是一套针对Java开发的微服务框架,该框架提供了一系列功能和组件,用于帮助开发者更轻松地构建和管理微服务架构。 该框架的源码是指该框架的具体实现代码,通过阅读源码,开发者可以深入了解框架的设计思想和实现细节,有助于更好地理解和使用该框架。 javaweb_cloud_elevue_pro微服务框架旗舰版源码的特点主要包括以下几点: 1. 模块化设计:源码采用模块化设计,将整个框架拆分成多个独立的模块,每个模块负责不同的功能,有利于代码的组织和维护。 2. 弹性扩展:框架采用了微服务架构,可以很方便地进行系统扩展,支持动态添加和移除服务实例,以适应不同的业务需求。 3. 高可用性:框架提供了集群和负载均衡的支持,可以提高系统的可用性和性能。 4. 分布式事务管理:源码包含了分布式事务管理的实现,可以保证系统在多节点环境下的数据一致性。 5. 可靠消息传递:框架提供了消息队列和分布式任务调度的功能,可以确保消息的可靠传递和任务的可靠执行。 6. 监控和管理:框架提供了丰富的监控和管理功能,可以对系统进行实时监控和管理,方便开发者进行故障排查和性能调优。 总之,javaweb_cloud_elevue_pro微服务框架旗舰版源码是一套成熟的微服务框架实现代码,通过研究源码,开发者可以更好地理解和使用该框架,提高开发效率和系统稳定性。 ### 回答2: javaweb_cloud_elevue_pro微服务框架是一款功能强大的微服务开发框架,是javaweb_cloud_elevue_pro产品家族中的旗舰版。 该框架的源码提供了一套完整的微服务解决方案,包含了丰富的功能模块和技术组件,可用于构建高性能、稳定可靠的分布式系统。 源码中包含了一系列核心组件,如服务注册与发现、负载均衡、熔断降级、分布式配置中心等,这些组件可以快速搭建起一个具有高可用性和可伸缩性的微服务架构。 此外,框架的源码还提供了一套易用的开发工具和开发规范,使开发者能够快速上手并高效地开发微服务应用。开发者可以根据自己的业务需求,选择合适的组件和模块进行使用,从而实现个性化的定制和扩展。 与其他微服务框架相比,javaweb_cloud_elevue_pro微服务框架的源码具有以下特点: 1. 高性能:框架使用了一系列优化的算法和技术,能够实现高效的请求处理和资源利用,保证系统的稳定和可靠。 2. 可扩展:源码提供了丰富的扩展接口和插件机制,开发者可以根据自己的需求进行个性化的扩展和定制。 3. 易用性:框架的源码结构清晰,文档详细,附带了示例和教程,使开发者能够快速理解和掌握框架的使用方法。 4. 社区支持:javaweb_cloud_elevue_pro框架拥有一个活跃的社区,开发者可以在社区中获取帮助、交流经验,并参与框架的改进和贡献。 总之,javaweb_cloud_elevue_pro微服务框架旗舰版源码提供了一个完整的微服务解决方案和一套易用的开发工具,可帮助开发者快速构建高性能、稳定可靠的分布式系统。它是微服务领域中的佼佼者,值得开发者深入研究和使用。 ### 回答3: javaweb_cloud_elevue_pro微服务框架旗舰版源码是一套用于构建微服务架构的开源代码。该框架旨在提供一种简单、高效的方式来搭建和管理分布式系统。 该框架基于Java语言,采用了云计算、分布式计算和微服务架构的设计理念,提供了一套完整的解决方案来构建和管理分布式应用。 该框架具有以下特点: 1. 架构灵活:框架提供了一套灵活的架构设计,可以根据业务需求进行扩展和定制。 2. 模块化设计:框架采用模块化设计,各个微服务模块之间可以独立部署和调用,实现了高内聚、低耦合的分布式系统。 3. 高可用性:框架支持负载均衡、容错和故障恢复等机制,保证了系统的高可用性和容错性。 4. 弹性扩展:框架支持水平扩展,可以根据需求动态增加或减少服务实例,实现了系统的弹性扩展和负载均衡。 5. 服务治理:框架提供了服务注册与发现、负载均衡、熔断器等功能,实现了对微服务的有效治理和管理。 6. 高性能:框架采用了高性能、高并发的设计和优化策略,提升了系统的性能和扩展能力。 总体而言,javaweb_cloud_elevue_pro微服务框架旗舰版源码是一套功能强大、灵活可扩展的微服务架构代码,可帮助开发人员快速构建和管理分布式系统。无论是在企业级应用程序还是在云计算环境中,该框架都能提供优秀的支持,并具有较高的性能和可靠性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Y_cen

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值