从零开始读RocketMq源码(一)生产者启动

目录

前言

获取源码

总概论

生产者实例

源码

A-01:设置生产者组名称

A-02:生产者服务启动

B-01:初始化状态

B-02:该方法再次对生产者组名称进行校验

B-03:判断是否为默认生产者组名称

B-04: 该方法是为了实例化MQClientInstance对象,mq客户端对象实例

B-05: 该方法就是将当前生产者对象注册到mqClientInstance中的producerTable集合中,并且生产者组名称作为key

B-06: 启动相关核心服务以及开启一系列定时任务(核心逻辑)

1. 开启请求-响应通道- this.mQClientAPIImpl.start();

2. 开启拉动式服务- this.mQClientAPIImpl.start();

3. 开启负载均衡服务- this.rebalanceService.start();

4. 开启推送服务- this.defaultMQProducer.getDefaultMQProducerImpl().start(false);

5. 启动各种计划任务- this.startScheduledTask();

a. 启动定时任务获取MQ注册中心nameServer的地址- MQClientInstance.this.mQClientAPIImpl.fetchNameServerAddr();

b. 定时从nameServer拉取topic信息到本地存储 -                                    MQClientInstance.this.updateTopicRouteInfoFromNameServer();

c. 定时清除离线的broker服务并给所有在线的broker发送心跳

d. 定时持久化消费偏移量数据

e. 定时调整消费者消息的线程池数量

B-07:初始化topic路由信息、topic訂閲信息以及topic端点映射信息

B-08:开启定时监测broker故障信息任务

B-09:发送心跳给所有的broker服务

B-10:开启定时扫描异步请求响应任务

A-03:开启监控和处理同步发送和异步发送操作的守护线程

A-04:开启消息轨迹和发送机制

总结

展望


前言

大概一年半前自己写了一篇《云原生》一文搞懂RocketMQ队列概述,这篇对rocketmq的相关概念和使用方法进行了整理概述,就像结尾说的都太局限于表面,简单使用还能将就,但一出现问题自己也很难排查,为了迈向技能的下一个阶段,还得是要读源码,学习大佬们的编码风格和技巧,对于使用mq以及排除问题也会更得心应手。因为最近一年自己的工作充满了波折,让自己没法静下心来学习整理,虽然现在也好不了多少,但可算能回归本心。本章篇幅比较长,将近万字,博主也是自己读源码一步一步跟踪的,所以尽量想描述得通俗易懂一些。

获取源码

首先我们从github上拉取rocketmqd的源码链接到本地,使用idea打开。

源码地址:https://github.com/apache/rocketmq

目前最新版本为:5.2.0

那么我们在idea上切换分支为 release-5.2.0

注:rocketmq5.x与4.x官方改动的东西比较多,尽量使用一直的版本,具体差异可查看官网,这里只对源码逻辑进行分析

总概论

我们知道rocketmq的组成需要四大模块构成,缺一不可

  • nameserver mq注册中心(状态管理)
  • broker mq的服务端(核心)
  • producer 生产者
  • consumer 消费者

本章我们先从应该大家接触最多的生产者开始学习源码吧。

生产者实例

  1. 在idea的rocketmq源码中找到 example 模块,这个模块中都是官方给出的简单案例
  2. 然后找到simple 包下面的 Producer类打开
  3. 然后在producer类中配置自己的mq的地址,topic以及tag就能成功启动生产者并且发送消息

注意:这里成功启动的前提是必须提前启动了mq的nameserver服务和broker服务才能成功,若没有可不用启动,直接跳过看下面源码

源码

根据上面简单生产者实例可知,生产者端的两大核心就是,启动生产者发送消息,分别对应下面两行代码。看似简单的两行其实里面的功能逻辑很强大。

  • producer.start();
    
  • producer.send(msg);

生产者包含4中状态:

  • CREATE_JUST 服务刚刚创建,尚未启动
  • RUNNING 服务运行中
  • SHUTDOWN_ALREADY 服务已关闭
  • START_FAILED 启动出错

按照顺序,我们从 生产者的启动开始

public void start() throws MQClientException {
    //A-01:设置生产者组名称
    this.setProducerGroup(withNamespace(this.producerGroup));
    //A-02:生产者服务启动
    this.defaultMQProducerImpl.start();
    //A-03:开启监控和处理同步发送和异步发送操作的守护线程
    if (this.produceAccumulator != null) {
        this.produceAccumulator.start();
    }
    //A-04:开启消息轨迹和发送机制
    if (null != traceDispatcher) {
        try {
            traceDispatcher.start(this.getNamesrvAddr(), this.getAccessChannel());
        } catch (MQClientException e) {
            logger.warn("trace dispatcher start failed ", e);
        }
    }
}

A-01:设置生产者组名称

  1. 该方法中顾名思义主要用于设置生产者组的名称
  2. withNamespace()进入该方法发现,其实对生产者组的名称就行各种非空校验和长度校验,最后根据固定格式拼接名称后返回。(对于开源组件大佬,校验方式也是和我们无异的)

A-02:生产者服务启动

该方法为本次的启动核心方法,我们直接深入了解下其内部实现。

方法逻辑太长我们进行分段拆分来解析

public void start(final boolean startFactory) throws MQClientException {
    switch (this.serviceState) {
            //B-01:初始化状态
        case CREATE_JUST:
            this.serviceState = ServiceState.START_FAILED;
            //B-02:校验
            this.checkConfig();
            //B-03:生产者组名设置
            if (!this.defaultMQProducer.getProducerGroup().equals(MixAll.CLIENT_INNER_PRODUCER_GROUP)) {
                this.defaultMQProducer.changeInstanceNameToPID();
            }
        //...
B-01:初始化状态

因为现在还是正在启动中,所以状态还是默认未启动状态,那么直接进入第一个case逻辑中,进入后里面把状态至为启动失败,我认为这是一种防御性编码,并且防止未成功启动的生产者被重复启动

B-02:该方法再次对生产者组名称进行校验
B-03:判断是否为默认生产者组名称

        前面可知我们已经成功设置自定义名称,所以直接进入if中

  • changeInstanceNameToPID(),该方法就设置实例名称,进入方法可以看到名称的生成规则,this.instanceName = UtilAll.getPid() + "#" + System.nanoTime();
    当前运行的虚拟机的名称截取拼接上当前纳米时间戳,保证唯一性
public void start(final boolean startFactory) throws MQClientException {
    /....../
    //B-04:该方法是为了实例化MQClientInstance对象,mq客户端对象实例
    this.mQClientFactory = MQClientManager.getInstance().getOrCreateMQClientInstance(this.defaultMQProducer, rpcHook);
    //B-05:注册生产者
    boolean registerOK = mQClientFactory.registerProducer(this.defaultMQProducer.getProducerGroup(), this);
    /....../
B-04: 该方法是为了实例化MQClientInstance对象,mq客户端对象实例
  • 内部首先生成一个唯一的clientId,其组成包含ip地址与之前生成的实例名称instanceName组成,然后new 了一个MQClientInstance对象并设置对应属性。
  • 将clientId作为key维护到一个Map对象中,private final ConcurrentMap<String/* clientId */, MQClientInstance> factoryTable;

注:MQClientInstance对象,该对象非常重要,因为生产者和消费者都在使用

进入该对象我们可以发现,里面维护了两个Map集合,就是分别存储当前客户端的生产者和消费者的对象数据

private final ConcurrentMap<String, MQProducerInner> producerTable
private final ConcurrentMap<String, MQConsumerInner> consumerTable

B-05: 该方法就是将当前生产者对象注册到mqClientInstance中的producerTable集合中,并且生产者组名称作为key
public void start(final boolean startFactory) throws MQClientException {
    switch (this.serviceState) {
            /..../
            //B-06: 启动相关核心服务以及开启一系列定时任务
            if (startFactory) {
                mQClientFactory.start();
            }
            /.../
B-06: 启动相关核心服务以及开启一系列定时任务(核心逻辑)
1. 开启请求-响应通道- this.mQClientAPIImpl.start();
2. 开启拉动式服务- this.mQClientAPIImpl.start();
3. 开启负载均衡服务- this.rebalanceService.start();
4. 开启推送服务- this.defaultMQProducer.getDefaultMQProducerImpl().start(false);
  • 这个方法是否有点眼熟,没错这就是我们最开始调用的启动方法A-2,参数传的false,说明上面if代码块中startFactory=false,则不进入B-06的代码块中
  • 并且A-2代码块方法中,因为第一次进入时状态已经从CREATE_JUST变更为START_FAILED,所以也不会再次进入第一个case中
  • 阅读后续代码可知,核心就是调用了 this.mQClientFactory.sendHeartbeatToAllBrokerWithLock(); 向所有Broker服务发送一次心跳(具体后面会详解)
5. 启动各种计划任务- this.startScheduledTask();

所有任务都是使用Executors线程池创建一个单独的的单线程定时任务实现,如下格式

private final ScheduledExecutorService scheduledExecutorService = Executors.newSingleThreadScheduledExecutor(r -> new Thread(r, "MQClientFactoryScheduledThread"));
//....
this.scheduledExecutorService.scheduleAtFixedRate(() -> {
    try {
        //业务逻辑
    } catch (Exception e) {
        log.error("ScheduledTask fetchNameServerAddr exception", e);
    }
}, 1000 * 10, 1000 * 60 * 2, TimeUnit.MILLISECONDS);
a. 启动定时任务获取MQ注册中心nameServer的地址- MQClientInstance.this.mQClientAPIImpl.fetchNameServerAddr();

首次启动延迟时间:2s

定时间隔时间:2m

mQClientAPIImpl对象是否眼熟,没错就是上面B-06-1启动的服务,所以该服务必须在任务执行之前启动,查看源码如此

  • 深入方法中会发现其实就是获取地址处理后存储在一个List集合中,为什么使用集合,我认为如果是集群那就就会有多条地址存在。 private final AtomicReference<List<String>> namesrvAddrList = new AtomicReference<>();
  • 继续深入会发现有Netty的身影,用于服务间远程通信,这里不再研究。
  • private final ConcurrentMap<String /* addr */, ChannelWrapper> channelTables;
  • 该Map就是用nameserver地址作为key,而value为ChannelWrapper对象,该对象内部就使用了netty框架 包中的对象,一个地址对应一个通道封装器。但是该逻辑中并没有使用put操作,只是get获取。
b. 定时从nameServer拉取topic信息到本地存储 -                                    MQClientInstance.this.updateTopicRouteInfoFromNameServer();

首次启动延迟时间:10ms

定时间隔时间:30s

  • 深入方法内部可知,其实就是分别对producerTableconsumerTable的map进行操作遍历,取出对象里面的topic名称,由前面B-04中可知,分别用于存储生产者对象消费者对象信息
  • 再将topic名称的set集合进行遍历去远程获取nameserver中的topic的路由详细信息,并将信息存储在另一个map对象中。作用: 用于管理和查询主题的路由信息,帮助生产者和消费者确定消息的发送和接收路径。
  • private final ConcurrentMap<String/* Topic */, TopicRouteData> topicRouteTable = new ConcurrentHashMap<>();
c. 定时清除离线的broker服务并给所有在线的broker发送心跳

        MQClientInstance.this.cleanOfflineBroker(); 清除离线的broker

        MQClientInstance.this.sendHeartbeatToAllBrokerWithLock(); 给所有的broker发送心跳

首次启动延迟时间:1s

定时间隔时间:30s

  • 清除离线的broker,查看源码可知道,大概意思为首先从private final ConcurrentMap<String, HashMap<Long, String>> brokerAddrTable = new ConcurrentHashMap<>(); map中
    • 获取所有broker的地址数据,然后进行遍历,
    • 在遍历中取出 topicRouteTable,该map存放的是topic的对象信息
    • 再对topic map的values进行遍历,取出topic信息对象中存储的对应broker集合,
    • 判断上面的brokerAddrTable中的broker是否在topic维护的broker集合中,没有则清除
d. 定时持久化消费偏移量数据

     MQClientInstance.this.persistAllConsumerOffset();

首次启动延迟时间:10s

定时间隔时间:5s

同样的维护了一个Map对象:

private ConcurrentMap<MessageQueue, ControllableOffset> offsetTable;

key则为消息队列对象

  • 深入源码可知,它的消费者持久化实现方式有三种
    • lite pull
    • mp pull
    • mp push
e. 定时调整消费者消息的线程池数量

    MQClientInstance.this.adjustThreadPool();

首次启动延迟时间:1m

定时间隔时间:1m

public void start(final boolean startFactory) throws MQClientException {
    switch (this.serviceState) {
            //...
            //B-07:初始化topic路由信息、topic訂閲信息以及topic端点映射信息
            this.initTopicRoute();
            //B-08:开启定时监测broker故障信息任务
            this.mqFaultStrategy.startDetector();
            //...
B-07:初始化topic路由信息、topic訂閲信息以及topic端点映射信息
  • 深入源码可知,首先获取开发者自定义的topic集合,然后分别处理成MQ要求的格式newTopic,然后创建TopicPublishInfo对象,用于存储topic订阅信息newTopic作为key,同样最后放入map中

private final ConcurrentMap<String/* topic */, TopicPublishInfo> topicPublishInfoTable;

  • 查看TopicPublishInfo对象可知,对象里面包含了TopicRouteData对象,我们知道这个对象在上面定时器B-06-5-b中出现过用于存储topic路由信息,并且存储在topicRouteTable map中
  • 所以在本方法中也会通过newTopic去远程从nameserver中拉去TopicRouteData信息,设置到TopicPublishInfo对象中,同样也会对比topic新获取的TopicRouteData与原来定时器存储的topicRouteTable中的是否有变化,有则更新
  • 有变化同时还会更新,上面定时器B-06-5-c中出现的brokerAddrTable map,更新broker的地址信息
  • 同时更新topic端点映射信息-记录每个主题的消息队列与 Broker 之间的映射 
  • private final ConcurrentMap<String/* Topic */, ConcurrentMap<MessageQueue, String/*brokerName*/>> topicEndPointsTable;这是一个嵌套map,因为一个topic可能对应多个broker,那么消息队列也会是对应多个broker, 可以帮助管理和均衡负载,确保消息被分布到不同的 Broker 上
B-08:开启定时监测broker故障信息任务

深入源码可知,里面维护了一个定时任务,定时监测 Broker 的故障详细信息

首次启动延迟时间:3s

定时间隔时间:3s

  • 同时也维护了一个map,用于存储每一个broker的 故障详细信息,包括故障时间、故障持续时间和可用状态

        private final ConcurrentHashMap<String, FaultItem> faultItemTable;

  • 逻辑处理中还会去查询brokerAddrTable中是否还存在当前broker地址信息,不存在则从faultItemTable中移除,然后再去监测broker服务是否可用,若可用则将可用状态 设置为true
public void start(final boolean startFactory) throws MQClientException {
    //...
    //B-09:发送心跳给所有的broker服务
    this.mQClientFactory.sendHeartbeatToAllBrokerWithLock();
    //B-10:开启定时扫描异步请求响应任务
    RequestFutureHolder.getInstance().startScheduledTask(this);

}
B-09:发送心跳给所有的broker服务

        发送心跳其实在上面定时器B-06-5-c中已经出现过了,但是没有深入了解,那么定时器中既然已经在发送心跳了,为什么生产者启动最后还要发送呢?

  • 定时任务的作用:定时任务确保客户端在运行过程中定期发送心跳,保持与 Broker 的连接。
  • 启动时的心跳:生产者在启动完成时立即发送心跳,以确保初始化成功、快速检测连接状态并更新路由信息。
  • 同样的心跳机制中也维护了一个map, 用于记录和管理每个 Broker 的心跳信息,private final ConcurrentMap<String, Integer> brokerAddrHeartbeatFingerprintTable;
  • 其中value值称为心跳指纹, MQ通过比较当前心跳指纹和上次记录的指纹,可以判断 Broker 是否正常工作
B-10:开启定时扫描异步请求响应任务

        深入源码可知,里面维护了一个定时任务,定时扫描MQ存储的生产者发布的异步请求以及响应的信息,帮助MQ实现异步请求的超时、回调和状态管理,增强系统的异步处理能力。

次启动延迟时间:3s

定时间隔时间:1s

同样的是维护了一个map数据用于存储异步请求以及响应的信息:

private ConcurrentHashMap<String, RequestResponseFuture> requestFutureTable

那么key为请求时生成的唯一标识,value为RequestResponseFuture对象则记录了请求信息、超时时间、响应信息、回调信息等,mq根据记录的信息做出响应的处理。

  • 源码内部逻辑有一个地方就判断了isTimeout是否请求超时,为true则抛出异常

该map requestFutureTable在本次启动中只是使用,具体在什么地方存储的,应该会在后续的生产者发送消息源码中再次出现,本次启动使用到的requestFutureTable应该都是没数据的。日常开发看似只是简单的调用了发送消息的api方法,而mq内部则做了许多复杂的处理来保证消息的可靠性和高可用性

A-03:开启监控和处理同步发送和异步发送操作的守护线程

  • guardThreadForSyncSend.start();
  • guardThreadForAsyncSend.start();

        这些线程中,可以实现具体的监控和处理逻辑,例如检测发送超时、重试失败的发送操作等。 并且这些线程在 JVM 退出时会自动终止

A-04:开启消息轨迹和发送机制

通过收集消息轨迹信息,可以了解消息在 RocketMQ 中的流转路径,帮助系统监控和故障排查。

总结

对于RocketMQ我们都知道生产者会从nameserver中拉取数据,并且会在本地存储,就算nameserver服务意外离线了,也能通过本地保存的数据进行消息通信。那么如何远程拉取数据以及心跳监测如何在本地存储,我想大家通过对上面start启动源码的学习,疑惑都解开了吧。

  • 数据更新以及心跳无非就是通过一系列的定时器在不断请远程请求
  • 本地存储则是使用已 table为后缀命名的Map集合来存储的

对本章源码中遇到的定时器和table进行了整理,方便大家快速记忆


展望

本章内容比较多,博主也是肝了几天才完成,希望对大家都有所收获,下一章我们继续对生产者send消息源码进行学习!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

零基础STM32单片机编程入门(八)定时器PWM输入实战含源码视频

文章目录 一.概要二.PWM输入框架图三.CubeMX配置一个PWM输入例程1.硬件准备2.创建工程3.调试 四.CubeMX工程源代码下载五.讲解视频链接地址六.小结 一.概要 脉冲宽度调制(PWM)&#xff0c;是英文“Pulse Width Modulation”的缩写&#xff0c;简称脉宽调制&#xff0c;是利用单…

转发服务器实验

首先先克隆一个虚拟机并完成ip地址的修改 nmcli connection modify ens160 ipv4.addresses 192.168.209.128/24 nmcli connection modify ens160 ipv4.method manual nmcli connection modify ens160 connection.autoconnect yes nmcli connection up ens160 nmcli connection…

计算机网络浅谈—什么是 OSI 模型?

开放系统通信&#xff08;OSI&#xff09;模型是一个代表网络通信工作方式的概念模型。 思维导图 什么是 OSI 模型&#xff1f; 开放系统互连 (OSI) 模型是由国际标准化组织创建的概念模型&#xff0c;支持各种通信系统使用标准协议进行通信。简单而言&#xff0c;OSI 为保证…

读书到底有什么意义?从笨小孩到名人的逆袭之路

点击上方△腾阳 关注 作者 l 腾阳 转载请联系授权 读书到底有什么意义&#xff1f; 有一个鸟语花香的农场里&#xff0c;住着老农夫和他的小孙子。 老农夫经常在清晨会坐在窗边&#xff0c;捧着厚厚的《圣经》&#xff0c;沉浸在知识的海洋里。 小孙子问他&#xff1a;…

【Linux】文件系统6——理解文件操作

目录 1.文件的读取 1.1.目录 1.2.文件 1.3.目录树读取 1.4.文件系统大小与磁盘读取性能 2.增添文件 2.1.数据的不一致&#xff08;Inconsistent&#xff09;状态 2.2.日志式文件系统&#xff08;Journaling filesystem&#xff09; 3.Linux文件系统的运行 4、文件的删…

Selenium的自动化测试技巧有多少?【建议收藏】

Selenium是一个用于自动化Web应用程序测试的工具。它提供了一组API&#xff0c;允许用户与Web浏览器进行交互&#xff0c;来执行各种自动化测试任务。本文将从零开始&#xff0c;详细介绍Selenium的自动化测试技巧。 第一步&#xff1a;安装Selenium 首先&#xff0c;您需要安…

【鸿蒙学习笔记】Stage模型工程目录

官方文档&#xff1a;应用配置文件概述&#xff08;Stage模型&#xff09; 目录标题 FA模型和Stage模型工程级目录模块级目录app.json5module.json5程序执行流程程序基本结构开发调试与发布流程 FA模型和Stage模型 工程级目录 模块级目录 app.json5 官方文档&#xff1a;app.j…

WAIC:生成式 AI 时代的到来,高通创新未来!

目录 01 在终端侧算力上&#xff0c;动作最快的就是高通 02 模型优化&#xff0c;完成最后一块拼图 在WAIC上&#xff0c;高通展示的生成式AI创新让我们看到了未来的曙光。 生成式 AI 的爆发带来了意想不到的产业格局变化&#xff0c;其速度之快令人惊叹。 仅在一个月前&…

android之蓝牙遥控器新增键值

文章目录 简述连接蓝牙代码流程总结简述 使用android 10平台来适配蓝牙遥控器新增的键值 连接蓝牙 当使用遥控器与蓝牙进行配对成功后,就可以通过getevent获取蓝牙打印的信息,如下所示 其中000700a0是发送过来的协议(0007)和码值(00a0)的组合。0xfa是驱动定义好的值,如果…

AI智能歌曲创作源码系统 前后端分离 带完整的安装代码包以及搭建教程

系统概述 本AI智能歌曲创作源码系统集成了深度学习、自然语言处理和音乐理论&#xff0c;旨在通过用户输入的关键词、情感色彩或音乐片段&#xff0c;自动生成具有创意且风格多样的音乐作品。系统核心由两大部分构成&#xff1a;前端用户界面与后端音乐生成引擎。前端负责接收…

什么是 VueQuill(前端的富文本编辑器)?

什么是 VueQuill&#xff1f; 1. 简介 VueQuill 是 Vue.js 的一个富文本编辑器插件&#xff0c;它基于 Quill 编辑器构建&#xff0c;提供了简洁且功能强大的富文本编辑功能。Quill 是一个现代化的富文本编辑器&#xff0c;提供丰富的文本编辑能力&#xff0c;支持多种格式和…

【学习笔记】程序设计竞赛

程序设计竞赛 文章目录 程序设计竞赛0x00 基本操作指南0x01 算法分析0x02 STL和基本数据结构栈队列集合map 0x03 排序插入排序归并排序&#xff08;Merge Sort)快速排序 0x04 搜索技术BFSDFS回溯与剪枝 深度迭代ID A*A star双向广搜 0x05 递推方程0x06 高级数据结构并查集二叉树…

【强化学习的数学原理】课程笔记--3(蒙特卡洛方法)

目录 蒙特卡洛方法MC Basic算法sparse reward MC Greedy 算法样本使用效率MC ϵ \epsilon ϵ-Greedy 算法一些例子 蒙特卡洛方法 第二节 推导贝尔曼最优公式中的&#xff1a; q π k ( s , a ) ∑ r P ( r ∣ s , a ) r γ ∑ s ′ P ( s ′ ∣ s , a ) v π k ( s ′ ) q…

CMS Made Simple v2.2.15 远程命令执行漏洞(CVE-2022-23906)

前言 CVE-2022-23906 是一个远程命令执行&#xff08;RCE&#xff09;漏洞&#xff0c;存在于 CMS Made Simple v2.2.15 中。该漏洞通过上传头像功能进行利用&#xff0c;攻击者可以上传一个经过特殊构造的图片文件来触发漏洞。 漏洞详情 CMS Made Simple v2.2.15 中的头像上…

NAS 必备导航页 Homepage 外观简约但功能丰富

本文首发于只抄博客,欢迎点击原文链接了解更多内容。 前言 NAS 上的应用部署多了之后,不同的服务对应的端口很难记住,在内网中使用,一般也不会绑定域名。 此时就需要有一个导航页将 NAS 上部署的所有服务都罗列出来,方便我们直接点击访问对应的服务。 今天给大家介绍的…

电商利器——淘宝商品月销量API接口解析

在电商时代&#xff0c;数据就是金钱。对于淘宝商家而言&#xff0c;掌握商品的销量数据无异于掌握了市场的脉搏。如今&#xff0c;淘宝商品月销量API接口的出现&#xff0c;联讯数据让商家如虎添翼&#xff0c;能够更加精准地把握市场动态&#xff0c;优化商品策略。 淘宝商…

强技能 展风采 促提升——北京市大兴区餐饮行业职工技能竞赛精彩呈现

6月19日&#xff0c;由大兴区总工会、区商务局、青云店镇人民政府联合主办&#xff0c;区服务工会、区餐饮行业协会承办的“传承中国技艺&#xff0c;打造新一代餐饮工匠”2024年大兴区餐饮行业职工职业技能竞赛决赛在北京华联创新学习中心隆重开幕。区总工会副主席郝泽宏&…

力扣5----最长回文子串

给你一个字符串 s&#xff0c;找到 s 中最长的回文子串 示例 1&#xff1a; 输入&#xff1a;s "babad" 输出&#xff1a;"bab" 解释&#xff1a;"aba" 同样是符合题意的答案。示例 2&#xff1a; 输入&#xff1a;s "cbbd" 输出…

均匀采样信号的鲁棒Savistky-Golay滤波(MATLAB)

S-G滤波器又称S-G卷积平滑器&#xff0c;它是一种特殊的低通滤波器&#xff0c;用来平滑噪声数据。该滤波器被广泛地运用于信号去噪&#xff0c;采用在时域内基于多项式最小二乘法及窗口移动实现最佳拟合的方法。与通常的滤波器要经过时域&#xff0d;频域&#xff0d;时域变换…

Docker:Docker网络

Docker Network 是 Docker 平台中的一项功能&#xff0c;允许容器相互通信以及与外界通信。它提供了一种在 Docker 环境中创建和管理虚拟网络的方法。Docker 网络使容器能够连接到一个或多个网络&#xff0c;从而使它们能够安全地共享信息和资源。 预备知识 推荐先看视频先有…