微服务实用篇4-消息队列MQ

news2024/11/25 10:37:18

今天主要来学习异步通讯技术MQ,主要包括初识MQ,RabbitMQ快速入门,SpringAMQP三大部分,下面就来一起学习吧。路漫漫其修远兮,吾将上下而求索,继续加油吧,少年。

目录

一、初识MQ

1.1、同步通讯和异步通讯的优缺点

1.2、MQ常见技术介绍

 1.3、RabbitMQ介绍与安装

1.4、RabbitMQ消息队列模型

二、SpringAMQP

2.1、基本介绍

2.2、SpringAMQP入门案例之消息发送

2.3、SpringAMQP入门案例之消息接收

2.4、SpringAMQP之工作队列Work Queue

2.5、SpringAMQP之发布-订阅模型广播交换机

2.6、SpringAMQP之发布-订阅模型路由交换机

2.7、SpringAMQP之发布-订阅模型主题交换机

2.8、SpringAMQP之消息转换器


一、初识MQ

1.1、同步通讯和异步通讯的优缺点

我们先看一下同步调用的优缺点,同步调用是实时响应,可以立即得到结果。但是同步调用一般耦合度较高,性能偏低,还存在级联失败等问题。

下面我们看一下微服务之间的异步调用,异步调用主要是解决了同步调用存在的一些问题,异步调用的优点:发布订阅的模式,耦合度低,不不要等待,吞吐量高,故障隔离,不会出现级联失败的问题,流量销峰,broker缓存,微服务根据自己的能力从broker 中获取。

1.2、MQ常见技术介绍

MQ即Message Queue,消息队列,就是存放消息的队列,也就是事件驱动架构中的Broker,常用的四种消息队列:RabbitMQ,ActiveMQ,RocketMQ,Kafka。对于稳定性要求较高的情况下,一般使用RabbitMQ或RocketMQ,对于数据量比较大,性能要求比较高的一般用Kafka。

 1.3、RabbitMQ介绍与安装

我们先看一下RabbitMQ的架构,首先发布者发布消息到交换即,交换即通过哦队列进行缓存消息,最后消费者通过订阅从队列中取消息。

我们先了解一下RabbitMQ中的一些概念,channel用于操作MQ,exchange是交换机,用来路由消息到队列中,队列queue用于缓存消息,virtual host是虚拟主机,用于对队列和交换即等进行逻辑分组。

下面我们进行RabbitMQ的安装过程如下,我们在centos7虚拟机中使用docker进行安装。

有两种方式下载MQ,第一种是在线拉取镜像包,如下:

docker pull rabbitmq:3-management

第二种,下载方式,是本地已经有了镜像包,通过Xftp进行上传到虚拟机,然后使用dokcer命令加载镜像即可。

docker load -i mq.tar

镜像加载完成后,可以使用docker images进行查看,具体如下,可以发现MQ镜像导入成功:

加载好MQ的镜像后,就需要使用指令进行安装,指令如下:

该指令设置了MQ的用户名和密码变量,设置名称和主机名,设置两个端口,一个MQ管理平台的端口,另外一个是作消息通信的一个端口。

docker run \
-e RABBITMQ_DEFAULT_USER=root \
-e RABBITMQ_DEFAULT_PASS=123456 \
--name mq \
--hostname mq1 \
-p 15672:15672 \
-p 5672:5672 \
-d \
rabbitmq:3-management

安装完成后,通过设置的用户名和密码进行MQ的管理页面,如下:

1.4、RabbitMQ消息队列模型

RabbitMQ常见的5种消息队列模型如下,主要分为两大类,第一类是基本消息队列和工作消息队列,不包含交换机,第二种是发布订阅模式的,根据交换机的不同,分为三种类型:广播、路由和主题。

下面先来看一下简单的消息队列模型,只包括三个部分,即发布者、消息队列、订阅者。

下面总结一下基本消息队列的发送和接收流程,基本消息的发送首先需要建立连接,然后建立channel通道,利用通道进行声明队列和消息发送,将消息发送到队列中;基本消息队列的接收流程为:建立连接,创建通道,利用通道声明队列,先定义消费者的消费行为,然后利用通道channel将消费者与队列绑定,消费者就可以消费队列中的消息了。

二、SpringAMQP

2.1、基本介绍

我们先看一下什么是AMQP,AMQP是高级消息队列协议,是用于应用程序或传递业务消息的开放标准,SpringAMQP是Spring基于AMQP协议的一套API规范。

SpringAMQP的官方地址如下:https://spring.io/projects/spring-amqp

我们进入官方业面查看,可以发现,该项目主要包含两个部分,一个是基础抽象,另一个是RabbitAQ的基础实现。

可以发现amqp主要包含三个特征,第一个是用于异步处理消息的侦听器容器, 用于接收和发送给消息的RabbitTemplate,用于自动声明队列以及交换和绑定的RabbitAdmin。

2.2、SpringAMQP入门案例之消息发送

先看一下发送消息的基本案例,具体如如下,首先引入amqp依赖,然后利用RabbitTemplate对象发送消息到队列,在消费者中绑定该队列。

首先需要引入amqp的依赖,具体如下:

 <!--AMQP依赖,包含RabbitMQ-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-amqp</artifactId>
        </dependency>

在发布者中创建一个类,用于创建队列和消息并把消息发送到队列中。

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
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()
public class SpringTests {
    @Autowired
    private RabbitTemplate rabbitTemplate ;

    @Test
    public void testSend(){
        String queueName = "simple.queue" ;
        String message = "Spring amqp" ;
        rabbitTemplate.convertAndSend(queueName,message);
    }

}

当然需要配置MQ的ip地址和端口号,以及登录名和密码,之前设置的。

logging:
  pattern:
    dateformat: MM-dd HH:mm:ss:SSS
spring:
  rabbitmq:
    host: 192.168.102.130 # rabbitMQ的ip地址
    port: 5672 # 端口
    username: root
    password: 123456
    virtual-host: /

可以在浏览器的rabbitMQ管理界面查看到发送到队列的消息,如下所示。

 总结一下,SpringAMQP发送消息的流程,具体如下:

 

2.3、SpringAMQP入门案例之消息接收

首引入starter依赖,指定MQ地址,然后定义类使用@Component注解成bean交给Spring管理,然后在类的方法中@RabbitListener去监听消息队列,并获取消息。

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

import java.time.LocalTime;
import java.util.Map;

@Component
public class SpringRabbitListener {

    @RabbitListener(queues = "simple.queue")
    public void listenSimpleQueue(String msg) {
        System.out.println("消费者接收到simple.queue的消息:【" + msg + "】");
    }
}

2.4、SpringAMQP之工作队列Work Queue

工作队列模型有两个消费者,可以很好低提高消息处理的速度,避免队列消息堆积。

我们看下面的work queue的案例,实现一个队列绑定两个消费者。

 设置publisher发送消息,每20ms发送一次,1s发送50次消息,具体如下:

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
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()
public class SpringTests {
    @Autowired
    private RabbitTemplate rabbitTemplate ;

    @Test
    public void testWorkQueueSend() throws InterruptedException {
        String queueName = "simple.queue" ;
        String message = "hello message" ;
        for(int i=0; i<50; i++){ //1s发送50次
            rabbitTemplate.convertAndSend(queueName,message+i);
            Thread.sleep(20); //每发送一次消息,间隔20ms
        }

    }


}

使用两个消费者来消费消息,通过设置睡眠时间模拟消费者的消费能力,如下:

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

import java.time.LocalTime;
import java.util.Map;

@Component
public class SpringRabbitListener {
/*
    @RabbitListener(queues = "simple.queue")
    public void listenSimpleQueue(String msg) {
        System.out.println("消费者接收到simple.queue的消息:【" + msg + "】");
    }*/

    @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);
    }
}

对于消费者,配置prefetch可以设置每次只能取一个进行消费,消费完成再取,防止消费能力弱的消费者一次取多个,导致性能差。

logging:
  pattern:
    dateformat: MM-dd HH:mm:ss:SSS
spring:
  rabbitmq:
    host: 192.168.102.131 # rabbitMQ的ip地址
    port: 5672 # 端口
    username: root
    password: 123456
    virtual-host: /
    listener:
      simple:
        prefetch: 1

2.5、SpringAMQP之发布-订阅模型广播交换机

发布订阅模型是允许将一个消息发送给多个消费者,通过交换机实现,三种常用的交换机分别为广播、路由和话题。不过需要注意的是交换机负责消息路由,而不存储消息,如果路由失败,则消息丢失。

我们先看第一种发布订阅模式,广播的方式,交换机将收到的消息路由给每个与其绑定的队列。

我们使用SpringAMQP实现广播交换机的案例,具体如下:

即将一个消费者绑定到两个队列,如下:

import org.springframework.amqp.core.Binding;
import org.springframework.amqp.core.BindingBuilder;
import org.springframework.amqp.core.FanoutExchange;
import org.springframework.amqp.core.Queue;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class FanoutConfig {
    //声明广播类型的交换机1
    @Bean
    public FanoutExchange fanoutExchange(){
        return new FanoutExchange("fanout");
    }

    //声明队列1
    @Bean
    public Queue fanoutQueue1(){
        return new Queue("fanout.queue1");
    }

    // 绑定队列1到交换机
    @Bean
    public Binding fanoutBinding1(Queue fanoutQueue1, FanoutExchange fanoutExchange){
        return BindingBuilder
                .bind(fanoutQueue1)
                .to(fanoutExchange);
    }

    // 声明队列2
    @Bean
    public Queue fanoutQueue2(){
        return new Queue("fanout.queue2");
    }

    // 绑定队列2到交换机
    @Bean
    public Binding fanoutBinding2(Queue fanoutQueue2, FanoutExchange fanoutExchange){
        return BindingBuilder
                .bind(fanoutQueue2)
                .to(fanoutExchange);
    }

    @Bean
    public Queue objectQueue(){
        return new Queue("object.queue");
    }
}

在消费者的监听类中监听两个队列的消息,获取可以获取消息进行消费,如下:

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

import java.time.LocalTime;

@Component
public class SpringRabbitListener {


    @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);
    }

}

 最后,将消息发送到交换机,交换机会把消息路由给队列,这个交换是广播交换机,即广播给每个队列,如下:

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
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()
public class SpringTests {
    @Autowired
    private RabbitTemplate rabbitTemplate ;


    @Test
    public void testExchange1(){
        //交换机名称
        String exchangeName = "fanout" ;
        //消息
        String message = "hello" ;
        rabbitTemplate.convertAndSend(exchangeName,"",message);
    }

}

2.6、SpringAMQP之发布-订阅模型路由交换机

我们看一下路由交换机,这个的模式是根据规则将消息路由到指定的队列,即通过对比key的方式进行路由消息到相应的队列。

 

需要先监听两个队列,然后 绑定队列到交换机,为每个队列设置相应的key。

       @RabbitListener(queues = "fanout.queue1")
        public void listenFanoutQueue1(String msg) {
            System.out.println("消费者接收到fanout.queue1的消息:【" + msg + "】");
        }
        @RabbitListener(queues = "fanout.queue2")
        public void listenFanoutQueue2(String msg) {
            System.out.println("消费者接收到fanout.queue2的消息:【" + msg + "】");
        }

        @RabbitListener(bindings = @QueueBinding(
                value = @Queue(name = "direct.queue1"),
                exchange = @Exchange(name = "itcast.direct", type = ExchangeTypes.DIRECT),
                key = {"red", "blue"}
        ))
        public void listenDirectQueue1(String msg){
            System.out.println("消费者接收到direct.queue1的消息:【" + 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){
            System.out.println("消费者接收到direct.queue2的消息:【" + msg + "】");
        }

然后发送响应的消息到交换机就可以,需要指定key,通过对比key路由到指定的消息队列。


    @Test
    public void testExchange2(){
        //交换机名称
        String exchangeName = "itcast.direct" ;
        //消息
        String message = "hello, blue" ;
        rabbitTemplate.convertAndSend(exchangeName,"blue",message);
    }

2.7、SpringAMQP之发布-订阅模型主题交换机

下面看一下主题交换机和路由交换机的区别,主要就是绑定的方式不一样,主题交换机可以通过通配符的形式进行绑定,比较方便。

 首先需要在消费者的监听器中绑定交换机和队列,并使用通配符的模式指定key,如下:


        @RabbitListener(bindings = @QueueBinding(
                value = @Queue(name = "topic.queue1"),
                exchange = @Exchange(name = "itcast.topic", type = ExchangeTypes.TOPIC),
                key = "china.#"
        ))
        public void listenTopicQueue1(String msg){
            System.out.println("消费者接收到topic.queue1的消息:【" + msg + "】");
        }

        @RabbitListener(bindings = @QueueBinding(
                value = @Queue(name = "topic.queue2"),
                exchange = @Exchange(name = "itcast.topic", type = ExchangeTypes.TOPIC),
                key = "#.news"
        ))
        public void listenTopicQueue2(String msg){
            System.out.println("消费者接收到topic.queue2的消息:【" + msg + "】");
        }

按照key发送消息到交换机,交换机会路由到相应的队列,如下:
 

   @Test
    public void testExchange3(){
        //交换机名称
        String exchangeName = "itcast.topic" ;
        //消息
        String message = "大厂offer" ;
        rabbitTemplate.convertAndSend(exchangeName,"china.news",message);
    }

2.8、SpringAMQP之消息转换器

SpringAMQP会将对象序列化成字节后发送,然后进行反序列化即可接收。

 

首先引入依赖,json的依赖,如下:

  <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-databind</artifactId>
        </dependency>

在发布者和消费者的配置类中声明消息转换器,即序列化和反序列化,实现消息转换。

  @Bean
    public MessageConverter messageConverter(){
        return new Jackson2JsonMessageConverter();
    }

 剩下的就是发送消息,监听并消费消息即可。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/89266.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

文件历史记录无法识别此驱动器如何修复?

案例&#xff1a; 在电脑中尝试使用内置工具文件历史记录将文件备份到另一个硬盘时&#xff0c;发现如图所示的错误“文件历史记录无法识别此驱动器”&#xff0c;这可怎么办&#xff1f; 文件历史记录驱动器断开连接的原因 文件历史记录无法识别此驱动器的原因可能是启动类型…

四种排序(选择排序、冒泡排序、快速排序和插入排序)

四种排序&#xff08;选择排序、冒泡排序、快速排序和插入排序&#xff09;选择排序&#xff1a;完整代码&#xff1a;运行结果&#xff1a;冒泡排序&#xff1a;完整代码&#xff1a;运行结果&#xff1a;插入排序&#xff1a;完整代码&#xff1a;运行结果&#xff1a;快速排…

linux 环境异常登录的甄别方法

1、关于linux的登录记录 查看最近登录IP和历史命令执行日期 last 显示的最末尾的 使用last -10 看最新的 登录IP地址 时间 still仍在登录 选项&#xff1a; &#xff08;1&#xff09;-x&#xff1a;显示系统开关机以及执行等级信息 &#xff08;2&#xff09;-a&am…

SpringSecurity[3]-自定义登录逻辑,自定义登录页面,以及认证过程的其他配置

前一篇:SpringSecurity[2]-UserDetailsService详解以及PasswordEncoder密码解析器详解 链接:SpringSecurity[2]-UserDetailsService详解以及PasswordEncoder密码解析器详解_豆虫儿的博客-CSDN博客 五、自定义登录逻辑 当进行自定义登录逻辑时需要用到之前讲解的UserDetailsS…

Java泛型的作用以及如何使用(继承、接口、方法、通配符) 附源码

&#x1f34b;1. 泛型的定义 class 类名称<泛型标识, 泛型标识, ....>{private 泛型标识 变量名称;....... } 常用的泛型标识字符 :T, E, K, V E - Element (在集合中使用&#xff0c;由于集合中存放的是元素)&#xff0c;E是对各方法中的泛型类型进行限制&#xff0c;…

【微服务技术06】Nacos注册中心

【微服务技术06】Nacos注册中心 案例代码&#xff1a;https://gitee.com/pikachu2333/spring-cloud-hexuan 安装Nacos 安装nacos&#xff1a;https://github.com/alibaba/nacos/releases 单机启动&#xff1a;sh startup.sh -m standalone 访问&#xff1a;http://127.0.0.1…

操作系统,计算机网络,数据库刷题笔记10

操作系统&#xff0c;计算机网络&#xff0c;数据库刷题笔记10 2022找工作是学历、能力和运气的超强结合体&#xff0c;遇到寒冬&#xff0c;大厂不招人&#xff0c;可能很多算法学生都得去找开发&#xff0c;测开 测开的话&#xff0c;你就得学数据库&#xff0c;sql&#xf…

yolov5修改骨干网络-使用pytorch自带的网络-以Mobilenet和efficientnet为例

通过 yolov5修改骨干网络–原网络说明 我们知道&#xff1a;yolov5.yaml中存放的是我们模型构建参数&#xff0c;具体构建过程在yolo.py中的parse_model函数&#xff0c;通过循环遍历yolov5.yaml给的参数&#xff0c;去寻找网络名称&#xff0c;并将args的参数传入网络&#xf…

获取bean的三种方式和注意事项

获取bean的三种方式和注意事项 spring-ioc.xml <?xml version"1.0" encoding"UTF-8"?> <beans xmlns"http://www.springframework.org/schema/beans"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLoc…

PUMA:DOA估计模式的改进实现(Matlab代码实现)

&#x1f468;‍&#x1f393;个人主页&#xff1a;研学社的博客 &#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜…

北京旅游HTML学生网页设计作品 dreamweaver作业静态HTML网页设计模板 北京旅游景点网页作业制作 HTML+CSS+JS

&#x1f468;‍&#x1f393;学生HTML静态网页基础水平制作&#x1f469;‍&#x1f393;&#xff0c;页面排版干净简洁。使用HTMLCSS页面布局设计,web大学生网页设计作业源码&#xff0c;这是一个不错的旅游网页制作&#xff0c;画面精明&#xff0c;排版整洁&#xff0c;内容…

口诀速记 DataFrame 指定行列的数据

口诀速记DataFrame 指定行列的数据一、loc函数和iloc函数获取指定行列的数据二、loc和iloc的切片操作三、[] 语法获取指定行列的数据✅作者简介&#xff1a;大家好我是爱康代码 &#x1f4c3;个人主页&#xff1a;链接: 点一下这里就进入爱康代码的主页链接&#xff0c;欢迎参观…

深入:9种设计模式在Mybatis中的运用

虽然我们都知道有26个设计模式&#xff0c;但是大多停留在概念层面&#xff0c;真实开发中很少遇到&#xff0c;Mybatis源码中使用了大量的设计模式&#xff0c;阅读源码并观察设计模式在其中的应用&#xff0c;能够更深入的理解设计模式。 Mybatis至少遇到了以下的设计模式的…

接口优化技巧

1. 批量思想&#xff1a;批量操作数据库 优化前&#xff1a; //for循环单笔入库 for(TransDetail detail:transDetailList){insert(detail); }优化后&#xff1a; batchInsert(transDetailList);打个比喻&#xff1a; 打个比喻:假如你需要搬一万块砖到楼顶,你有一个电梯,电…

怎么进行多人配音?建议收藏这些方法

相信不少小伙伴平时在网上冲浪的时候&#xff0c;经常会刷到一些搞笑视频吧。这些搞笑视频的配音&#xff0c;通常都是以一人分饰多角或者是多人互动对话的形式进行配音的。那有没有小伙伴有着不错的搞笑创意点子&#xff0c;但是苦于没有人配音呢&#xff1f;其实我们可以使用…

电脑之间通信的大致过程

本文来自对网络工程师之路内容的个人总结&#xff0c;仅供个人复习参考。 1.电脑之间通信就需要有线路&#xff0c;但是如果多台电脑互相之间需要通信&#xff0c;那么就需要有很多根线&#xff0c;每台电脑需要有多网卡&#xff0c;为了解决这个问题&#xff0c;(集线器)Hub就…

360度内环镜、内螺纹检测镜头、瓶盖检测镜头以及超中心镜头

用于孔洞状物体的360内部成像 提示&#xff1a; 内孔检测光学镜头 从外部检查腔内&#xff1b;无需在孔洞内放置光学探头&#xff1b;带孔对象的360度对焦&#xff1b;腔体内壁和底部都可以实现高分辨率成像&#xff1b;景深可使同一个镜头拍摄具有不同形状和尺寸的物体&…

Bidirectional Recurrent Neural Networks

摘要 a regular recurrent neural network &#xff08;RNN&#xff09; →\rightarrow→ a bidirectional recurrent neural network (BRNN)a preset future frame&#xff1a; 预设的未来架构。. Structure and training procedure&#xff1a; 架构和训练程序。TIMIT datab…

Java多线程之:详解ThreadPoolExecutor执行源码分析

文章目录线程池的实现原理详解ThreadPoolExecutor核心数据结构核心配置参数解释线程池的优雅关闭线程池的生命周期正确关闭线程池的步骤shutdown()与shutdownNow()的区别任务的提交过程分析任务的执行过程分析shutdown()与任务执行过程综合分析shutdownNow() 与任务执行过程综合…

【大一大二必看】计算机专业的同学应该参加哪些比赛?

文章目录1. 前言2. ICPC3. CCPC4. 蓝桥杯5. 天梯赛6. CCF CSP7. PAT8. 全国高校计算机能力挑战赛9. 其他&#x1f351; 天池大赛&#x1f351; 华为软件精英挑战赛&#x1f351; LeetCode 周赛 / 双周赛&#x1f351; CSDN 编程竞赛总结1. 前言 2022 年已经过半&#xff0c;对…