当前位置: 首页 > news >正文

SpringBoot源码解析(十一):准备应用上下文

SpringBoot源码系列文章

SpringBoot源码解析(一):SpringApplication构造方法

SpringBoot源码解析(二):引导上下文DefaultBootstrapContext

SpringBoot源码解析(三):启动开始阶段

SpringBoot源码解析(四):解析应用参数args

SpringBoot源码解析(五):准备应用环境

SpringBoot源码解析(六):打印Banner

SpringBoot源码解析(七):应用上下文结构体系

SpringBoot源码解析(八):Bean工厂接口体系

SpringBoot源码解析(九):Bean定义接口体系

SpringBoot源码解析(十):应用上下文AnnotationConfigServletWebServerApplicationContext构造方法

SpringBoot源码解析(十一):准备应用上下文


目录

  • 前言
  • 源码入口
  • 一、执行应用上下文初始化器
    • 1、获取上下文初始化器
    • 2、执行初始化器
      • 2.1、DelegatingApplicationContextInitializer
      • 2.2、SharedMetadataReaderFactoryContextInitializer
      • 2.3、ContextIdApplicationContextInitializer
      • 2.4、ConfigurationWarningsApplicationContextInitializer
      • 2.5、RSocketPortInfoApplicationContextInitializer
      • 2.6、ServerPortInfoApplicationContextInitializer
      • 2.7、ConditionEvaluationReportLoggingListener
  • 二、触发应用监听器(上下文准备完成)
  • 三、关闭引导上下文
  • 四、触发应用监听器(上下文加载完成)
  • 总结

前言

  在前文中,我们介绍了应用上下文的构造方法初始化两个组件:注解Bean定义读取器和类路径Bean定义扫描器,接下来,我们将探究下准备应用上下文阶段对外的扩展点

SpringBoot版本2.7.18SpringApplication的run方法的执行逻辑如下,本文将详细介绍第8小节:刷新上下文

// SpringApplication类方法
public ConfigurableApplicationContext run(String... args) {// 记录应用启动的开始时间long startTime = System.nanoTime();// 1.创建引导上下文,用于管理应用启动时的依赖和资源DefaultBootstrapContext bootstrapContext = createBootstrapContext();ConfigurableApplicationContext context = null;// 配置无头模式属性,以支持在无图形环境下运行// 将系统属性 java.awt.headless 设置为 trueconfigureHeadlessProperty();// 2.获取Spring应用启动监听器,用于在应用启动的各个阶段执行自定义逻辑SpringApplicationRunListeners listeners = getRunListeners(args);// 启动开始方法(发布开始事件、通知应用监听器ApplicationListener)listeners.starting(bootstrapContext, this.mainApplicationClass);try {// 3.解析应用参数ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);// 4.准备应用环境,包括读取配置文件和设置环境变量ConfigurableEnvironment environment = prepareEnvironment(listeners, bootstrapContext, applicationArguments);// 配置是否忽略 BeanInfo,以加快启动速度configureIgnoreBeanInfo(environment);// 5.打印启动BannerBanner printedBanner = printBanner(environment);// 6.创建应用程序上下文context = createApplicationContext();// 设置应用启动的上下文,用于监控和管理启动过程context.setApplicationStartup(this.applicationStartup);// 7.准备应用上下文,包括加载配置、添加 Bean 等prepareContext(bootstrapContext, context, environment, listeners, applicationArguments, printedBanner);// 8.刷新上下文,完成 Bean 的加载和依赖注入refreshContext(context);// 9.刷新后的一些操作,如事件发布等afterRefresh(context, applicationArguments);// 计算启动应用程序的时间,并记录日志Duration timeTakenToStartup = Duration.ofNanos(System.nanoTime() - startTime);if (this.logStartupInfo) {new StartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(), timeTakenToStartup);}// 10.通知监听器应用启动完成listeners.started(context, timeTakenToStartup);// 11.调用应用程序中的 `CommandLineRunner` 或 `ApplicationRunner`,以便执行自定义的启动逻辑callRunners(context, applicationArguments);}catch (Throwable ex) {// 12.处理启动过程中发生的异常,并通知监听器handleRunFailure(context, ex, listeners);throw new IllegalStateException(ex);}try {// 13.计算应用启动完成至准备就绪的时间,并通知监听器Duration timeTakenToReady = Duration.ofNanos(System.nanoTime() - startTime);listeners.ready(context, timeTakenToReady);}catch (Throwable ex) {// 处理准备就绪过程中发生的异常handleRunFailure(context, ex, null);throw new IllegalStateException(ex);}// 返回已启动并准备就绪的应用上下文return context;
}

源码入口

// 7.准备应用上下文,包括加载配置、添加 Bean 等
prepareContext(bootstrapContext, context, environment, listeners, applicationArguments, printedBanner);
/*** 预处理应用上下文,在应用上下文刷新之前执行一系列准备工作。* * @param bootstrapContext  引导上下文,存储启动过程中的关键组件* @param context           Spring 应用上下文 AnnotationConfigServletWebServerApplicationContext * @param environment       Spring 运行环境* @param listeners         Spring 应用运行监听器* @param applicationArguments 应用程序运行参数* @param printedBanner     启动时显示的 Banner*/
private void prepareContext(DefaultBootstrapContext bootstrapContext, ConfigurableApplicationContext context,ConfigurableEnvironment environment, SpringApplicationRunListeners listeners,ApplicationArguments applicationArguments, Banner printedBanner) {// 设置应用环境context.setEnvironment(environment);// 对ApplicationContext进行处理(没啥重要内容,略过)postProcessApplicationContext(context);// 1. 执行应用程序上下文的初始化器applyInitializers(context);// 2. 触发 contextPrepared 事件,通知监听器上下文已准备好listeners.contextPrepared(context);// 3. 关闭引导上下文,因为之后不再需要bootstrapContext.close(context);// 记录应用启动信息(仅在日志启用时)if (this.logStartupInfo) {logStartupInfo(context.getParent() == null);logStartupProfileInfo(context);}// 获取 Bean 工厂,并注册特定的单例 Bean(后面章节注册Bean详解)ConfigurableListableBeanFactory beanFactory = context.getBeanFactory();// 注册命令行参数(使其可被 Spring 容器访问)beanFactory.registerSingleton("springApplicationArguments", applicationArguments);// 注册启动 Banner(如果存在)if (printedBanner != null) {beanFactory.registerSingleton("springBootBanner", printedBanner);}// 配置 Bean 工厂的属性if (beanFactory instanceof AbstractAutowireCapableBeanFactory) {// 允许/禁止循环依赖(Spring 5 改为默认禁用循环依赖)((AbstractAutowireCapableBeanFactory) beanFactory).setAllowCircularReferences(this.allowCircularReferences);// 允许/禁止 BeanDefinition 覆盖(默认禁止)if (beanFactory instanceof DefaultListableBeanFactory) {((DefaultListableBeanFactory) beanFactory).setAllowBeanDefinitionOverriding(this.allowBeanDefinitionOverriding);}}// 如果启用了懒加载,则添加Bean工厂后处理器(推迟 Bean 初始化)// 后面执行Bean工厂后置处理器时候详解if (this.lazyInitialization) {context.addBeanFactoryPostProcessor(new LazyInitializationBeanFactoryPostProcessor());}// 添加属性源排序后处理器,确保 `@PropertySource` 注解的顺序正确// @PropertySource 注解让你能够加载外部配置文件,并将配置文件中的属性注入到 Spring 的 Environment 中context.addBeanFactoryPostProcessor(new PropertySourceOrderingBeanFactoryPostProcessor(context));// 加载应用的所有源(配置类、XML 文件、组件扫描等)Set<Object> sources = getAllSources();Assert.notEmpty(sources, "Sources must not be empty"); // 这里就是将SpringBoot配置类注册为Bean定义放到bean工厂load(context, sources.toArray(new Object[0]));// 4. 触发 contextLoaded 事件,通知监听器上下文已加载完成listeners.contextLoaded(context);
}

一、执行应用上下文初始化器

protected void applyInitializers(ConfigurableApplicationContext context) {// 遍历所有需要应用到应用上下文的初始化器for (ApplicationContextInitializer initializer : getInitializers()) {// 通过 GenericTypeResolver 解析初始化器的泛型类型参数,确定所需的上下文类型Class<?> requiredType = GenericTypeResolver.resolveTypeArgument(initializer.getClass(),ApplicationContextInitializer.class);// 断言当前上下文的类型是否符合要求(指定类型、子类或实现类),如果不符合则抛出异常Assert.isInstanceOf(requiredType, context, "Unable to call initializer.");// 使用初始化器对应用上下文进行初始化initializer.initialize(context);}
}

1、获取上下文初始化器

  上下文初始化器是在SpringApplication构造方法中读取spring.factories文件获取的,这里就是将上下文初始化器集合拿来根据Ordered接口@Order排序

// SpringApplication构造方法中读取spring.factories文件获取的
private List<ApplicationContextInitializer<?>> initializers;// 获取应用上下文的初始化器集合
public Set<ApplicationContextInitializer<?>> getInitializers() {// 将初始化器集合转换为一个不可修改的、有序集合并返回return asUnmodifiableOrderedSet(this.initializers);
}// 将一个集合转换为一个不可修改的、有序的 Set
private static <E> Set<E> asUnmodifiableOrderedSet(Collection<E> elements) {// 将集合转换为列表,以便对其进行排序List<E> list = new ArrayList<>(elements);// 使用 AnnotationAwareOrderComparator 对列表进行排序// 通过Ordered接口或@Order排序list.sort(AnnotationAwareOrderComparator.INSTANCE);// 将排序后的列表转换为 LinkedHashSet,确保元素的顺序被保留return new LinkedHashSet<>(list);
}

debug看下获取到的初始化器

在这里插入图片描述

2、执行初始化器

  执行初始化器调用ApplicationContextInitializer实现类的initialize方法,下面逐一分析。

2.1、DelegatingApplicationContextInitializer

  简单描述一下,就是通过配置文件属性名context.initializer.classes获取到多个初始化器,然后再遍历调用每个初始化器的。

在这里插入图片描述

举例

假设在application.properties或通过命令行参数配置了以下内容:

context.initializer.classes=com.example.MyCustomInitializer,com.example.AnotherInitializer

  Spring在启动时会解析这个属性,并按照顺序实例化并调用 MyCustomInitializer 和 AnotherInitializer 类,这些类都应该实现ApplicationContextInitializer接口,负责在应用上下文初始化时进行自定义操作。

  DelegatingApplicationContextInitializer适用于需要在 Spring 应用程序启动时,按顺序动态加载执行多个不同的应用上下文初始化器的场景,特别是在具有多个模块或复杂配置的应用中。例如,当需要根据不同的环境配置(如开发、测试、生产环境)执行不同的初始化任务,或者在应用启动时灵活地执行自定义的初始化逻辑(如数据库连接配置、安全设置或外部服务的初始化)时,使用DelegatingApplicationContextInitializer可以帮助通过配置文件集中管理多个初始化器,确保它们按照指定的顺序正确执行。

2.2、SharedMetadataReaderFactoryContextInitializer

  SharedMetadataReaderFactoryContextInitializer 是 Spring 框架中的一个初始化器,它在 Spring 应用上下文初始化过程中,负责确保 MetadataReaderFactory 实例在整个上下文中共享,避免每次进行类路径扫描时都重新创建该工厂实例。MetadataReaderFactory主要用于读取和解析类的元数据,尤其是在注解扫描过程中,例如扫描 @Component, @Service, @Repository 等注解。通过共享MetadataReaderFactory,SharedMetadataReaderFactoryContextInitializer提高了类路径扫描的效率,减少了内存消耗和性能开销,尤其在大型或复杂的 Spring 应用中,能够加速启动过程。

2.3、ContextIdApplicationContextInitializer

  ContextIdApplicationContextInitializer 是 Spring 框架中的一个初始化器类,主要用于为 Spring 应用上下文生成一个唯一的 ID,并将其设置到应用上下文的环境中。这个 ID 通常用于标识不同的应用上下文实例,尤其在有多个 Spring 应用上下文存在的情况下,可以帮助区分它们。

2.4、ConfigurationWarningsApplicationContextInitializer

  ConfigurationWarningsApplicationContextInitializer 是 Spring 框架中的一个应用上下文初始化器,它用于在 Spring 应用程序启动时检查并发出有关配置方面的警告,尤其是关于弃用的配置潜在不推荐的配置。它的主要作用是在应用上下文初始化期间,帮助开发者识别不再推荐使用的配置或可能导致问题的配置方式。

2.5、RSocketPortInfoApplicationContextInitializer

  RSocketPortInfoApplicationContextInitializer 是一个 Spring 应用上下文初始化器,专门用于确保RSocket协议相关的端口信息在 Spring 应用启动时得到正确配置和初始化,特别适用于使用 RSocket 协议进行高效、双向通信的应用场景,如微服务架构和实时通信应用。

2.6、ServerPortInfoApplicationContextInitializer

  ServerPortInfoApplicationContextInitializer是一个确保在 Spring 启动时正确加载、处理和应用服务器端口信息的初始化器,它在处理动态端口配置、跨环境配置或微服务架构中的端口共享方面尤其有用。

2.7、ConditionEvaluationReportLoggingListener

  ConditionEvaluationReportLoggingListener 是 Spring Framework 中的一个类,用于在应用启动过程中记录和输出条件注解(如 @Conditional)的评估报告。它会将 Spring 配置类中的条件评估结果打印到日志中,帮助开发者了解哪些条件被满足,哪些未被满足,以及哪些 Bean 被加载或跳过

举例

如果在启动应用时使用 @Conditional 注解条件来控制 Bean 的加载,ConditionEvaluationReportLoggingListener 会在日志中输出类似如下的信息:

2018-12-04 16:42:52.755  INFO 12345 --- [  main] o.s.boot.autoconfigure.ConditionEvaluationReportLoggingListener : 
Evaluating conditions on com.example.SomeConfiguration- @ConditionalOnProperty (spring.datasource.url) matched => Spring DataSource Bean created- @ConditionalOnMissingBean (org.springframework.jdbc.datasource.DataSource) matched => Spring DataSource Bean created

二、触发应用监听器(上下文准备完成)

  listeners内部持有多个SpringApplicationRunListener(用于监听Spring应用程序启动过程的生命周期事件),这里在上下文准备完成时遍历所有SpringApplicationRunListener触发contextPrepared方法。

// 2. 触发 contextPrepared 事件,通知监听器上下文已准备好
listeners.contextPrepared(context);

在这里插入图片描述

  唯一的Spring应用启动监听器EventPublishingRunListener的上下文准备完成方法核心内容就是广播应用上下文初始化事件,将其推给合适的监听器(匹配监听器的事件类型,这里就是匹配上下文初始化事件的监听器)

在这里插入图片描述

虽然匹配到了如下应用监听器,但无操作内容在这里插入图片描述

三、关闭引导上下文

// 3. 关闭引导上下文,因为之后不再需要
bootstrapContext.close(context);

  主要内容就是发布事件给合适的监听器。目前没有此事件的监听器,故无任何操作。

// DefaultBootstrapContext类方法
public void close(ConfigurableApplicationContext applicationContext) {// 发布 BootstrapContextClosedEvent 事件,通知系统上下文已关闭this.events.multicastEvent(new BootstrapContextClosedEvent(this, applicationContext));
}

  BootstrapContextClosedEvent的发布和监听机制主要用于在应用程序上下文关闭时进行清理和通知。它适用于需要在应用关闭时执行清理操作、通知其他模块或外部系统、管理资源释放的场景。

四、触发应用监听器(上下文加载完成)

// 4. 触发 contextLoaded 事件,通知监听器上下文已加载完成
listeners.contextLoaded(context);

  核心内容与上下文准备完成触发的监听器原理一致,就是广播特定的事件到合适的监听器,这都是spring对外的扩展点,创建对应的事件监听器,就会在对应的时机触发执行。

在这里插入图片描述
如下匹配到的应用监听器,也没啥重要内容,略过

在这里插入图片描述

总结

  本文概述了SpringBoot启动过程的准备应用上下文阶段,触发上下文初始化器ApplicationContextInitializer、上下文准备完成ApplicationContextInitializedEvent事件监听器、引导上下文关闭BootstrapContextClosedEvent事件监听器、上下文加载完成ApplicationPreparedEvent事件监听器执行时机和流程,这些均可作为对外扩展点

相关文章:

SpringBoot源码解析(十一):准备应用上下文

SpringBoot源码系列文章 SpringBoot源码解析(一)&#xff1a;SpringApplication构造方法 SpringBoot源码解析(二)&#xff1a;引导上下文DefaultBootstrapContext SpringBoot源码解析(三)&#xff1a;启动开始阶段 SpringBoot源码解析(四)&#xff1a;解析应用参数args Sp…...

CSS 使用white-space属性换行

一、white-space属性的常见值 * 原本格式&#xff1a; 1、white-space:normal 默认值&#xff0c;空格和换行符会被忽略过滤掉&#xff1b;宽度不够时文本会自动换行 * 宽度足够时&#xff0c;normal 处理后的格式 * 宽度不够时&#xff0c; normal 处理后的格式 2、white-spa…...

论文笔记(七十二)Reward Centering(四)

Reward Centering&#xff08;四&#xff09; 文章概括摘要附录A 伪代码 文章概括 引用&#xff1a; article{naik2024reward,title{Reward Centering},author{Naik, Abhishek and Wan, Yi and Tomar, Manan and Sutton, Richard S},journal{arXiv preprint arXiv:2405.09999…...

Matlab——图像保存导出成好看的.pdf格式文件

点击图像的右上角&#xff0c;点击第一个保存按钮键。...

官方文档学习TArray容器

一.TArray中的元素相等 1.重载一下 元素中的 运算符&#xff0c;有时需要重载排序。接下来&#xff0c;我们将id 作为判断结构体的标识。 定义结构体 USTRUCT() struct FXGEqualStructInfo {GENERATED_USTRUCT_BODY() public:FXGEqualStructInfo(){};FXGEqualStructInfo(in…...

unxi-进程间通信

1.进程间通信实现方式 【1】同一主机 linux下通信方式: a.传统的进程间通信方式 管道 --- 进行数据传输的"管道" 无名管道 有名管道 信号 --- b.system v 进程间通信 (posix 进程间通信) 共享内存 (进程间…...

微型分组加密算法TEA、XTEA、XXTEA

微型分组加密算法TEA、XTEA、XXTEA TEA&#xff08;Tiny Encryption Algorithm&#xff09;算法是一种分组加密算法&#xff0c;由剑桥大学计算机实验室的‌David Wheeler和‌Roger Needham于1994年发明。TEA、XTEA、XXTEA算法采用64位的明文分组和128位的密钥。它使用Feistel…...

conda 基本命令

1、查询当前所有的环境 conda env list 2、创建虚拟环境 conda create -n 环境名 [pythonpython版本号] 其中[pythonpython版本号]可以不写 conda create -n test python3.12 我们输入conda env list看到我们的环境创建成功了&#xff0c;但是发现他是创建在我们默认的C盘的…...

详解 为什么 tcp 会出现 粘包 拆包 问题

TCP 会出现 粘包 和 拆包 问题&#xff0c;主要是因为 TCP 是 面向字节流 的协议&#xff0c;它不关心应用层发送的数据是否有边界&#xff0c;也不会自动分割或合并数据包。由于 TCP 的流控制和传输机制&#xff0c;数据可能在传输过程中被拆分成多个小的 TCP 包&#xff0c;或…...

Linus的基本命令

以下是一些常见的 Linux 命令&#xff1a; 一、文件和目录操作&#xff1a; - ls&#xff1a;列出目录中的文件和子目录&#xff0c;常用参数有 -a &#xff08;显示所有文件&#xff0c;包括隐藏文件&#xff09;、 -l &#xff08;显示详细信息&#xff09;、 -h &#xff0…...

【Linux】缓冲区和文件系统

个人主页~ 缓冲区和文件系统 一、FILE结构1、fd2、缓冲区&#xff08;一&#xff09;有换行有return全部打印&#xff08;二&#xff09;无换行无return的C接口打印&#xff08;三&#xff09;无换行无return的系统调用接口打印&#xff08;四&#xff09;有换行无return的C接口…...

函数式编程:概念、特性与应用

1. 函数式编程简介 函数式编程&#xff0c;从名称上看就与函数紧密相关。它是一种我们常常使用却可能并未意识到的编程范式&#xff0c;关注代码的结构组织&#xff0c;强调一个纯粹但在实际中有些理想化的不可变世界&#xff0c;涉及数学、方程和副作用等概念&#xff0c;甚至…...

git中的merge和rebase的区别

在 Git 中&#xff0c;git merge 和 git rebase 都是用于整合分支变更的核心命令&#xff0c;但它们的实现方式和结果有本质区别。以下是两者的详细对比&#xff1a; 一、核心区别 特性git mergegit rebase历史记录保留分支拓扑&#xff0c;生成新的合并提交线性化历史&#x…...

【目标检测】目标检测中的数据增强终极指南:从原理到实战,用Python解锁模型性能提升密码(附YOLOv5实战代码)

&#x1f9d1; 博主简介&#xff1a;曾任某智慧城市类企业算法总监&#xff0c;目前在美国市场的物流公司从事高级算法工程师一职&#xff0c;深耕人工智能领域&#xff0c;精通python数据挖掘、可视化、机器学习等&#xff0c;发表过AI相关的专利并多次在AI类比赛中获奖。CSDN…...

uniapp在app下使用mqtt协议!!!支持vue3

什么&#xff1f;打包空白&#xff1f;分享一下我的解决方法&#xff01; 第一步 找大师算过了&#xff0c;装4.1版本运气好&#xff01; 所以根目录执行命令… npm install mqtt4.1.0第二步 自己封装一个mqtt文件方便后期开坛做法&#xff01; // utils/mqtt.js import mqt…...

VMware虚拟机17.5.2版本下载与安装(详细图文教程包含安装包)

文章目录 前言一、vmware虚拟机下载二、vmware虚拟机安装教程三、vmware虚拟机许可证 前言 VMware Workstation Pro 17 功能强大&#xff0c;广受青睐。本教程将带你一步步完成它的安装&#xff0c;简单易上手&#xff0c;助你快速搭建使用环境。 一、vmware虚拟机下载 VMwar…...

如何加固织梦CMS安全,防webshell、防篡改、防劫持,提升DedeCMS漏洞防护能力

织梦系统&#xff08;DedeCMS&#xff09;是一款非常知名的CMS系统&#xff0c;因其功能强大、结构科学合理&#xff0c;深受广大用户喜欢。 虽然织梦CMS&#xff08;DedeCMS&#xff09;非常优秀&#xff0c;但是为了保障网站安全&#xff0c;我们还是需要做一些必要的防护措…...

STM32的HAL库开发---ADC采集内部温度传感器

一、STM32内部温度传感器简介 二、温度计算方法 F1系列&#xff1a; 从数据手册中可以找到V25和Avg_Slope F4、F7、H7系列只是标准值不同&#xff0c;自行查阅手册 三、实验简要 1、功能描述 通过ADC1通道16采集芯片内部温度传感器的电压&#xff0c;将电压值换算成温度后&…...

Linux 命令大全完整版(12)

Linux 命令大全 5. 文件管理命令 ln(link) 功能说明&#xff1a;连接文件或目录。语  法&#xff1a;ln [-bdfinsv][-S <字尾备份字符串>][-V <备份方式>][--help][--version][源文件或目录][目标文件或目录] 或 ln [-bdfinsv][-S <字尾备份字符串>][-V…...

Python - 代码片段分享 - Excel 数据实时写入方法

文章目录 前言注意事项工具 pandas1. 简介2. 安装方式3. 简单介绍几个api 实战片段 - 实时写入Excel文件结束语 要么出众&#xff0c;要么出局 前言 我们在爬虫采集过程中&#xff0c;总是将数据解析抓取后统一写入Excel表格文件&#xff0c;如果在解析数据出现问题容易出现数据…...

微软PowerBI考试 PL300-选择 Power BI 模型框架【附练习数据】

微软PowerBI考试 PL300-选择 Power BI 模型框架 20 多年来&#xff0c;Microsoft 持续对企业商业智能 (BI) 进行大量投资。 Azure Analysis Services (AAS) 和 SQL Server Analysis Services (SSAS) 基于无数企业使用的成熟的 BI 数据建模技术。 同样的技术也是 Power BI 数据…...

【python异步多线程】异步多线程爬虫代码示例

claude生成的python多线程、异步代码示例&#xff0c;模拟20个网页的爬取&#xff0c;每个网页假设要0.5-2秒完成。 代码 Python多线程爬虫教程 核心概念 多线程&#xff1a;允许程序同时执行多个任务&#xff0c;提高IO密集型任务&#xff08;如网络请求&#xff09;的效率…...

用docker来安装部署freeswitch记录

今天刚才测试一个callcenter的项目&#xff0c;所以尝试安装freeswitch 1、使用轩辕镜像 - 中国开发者首选的专业 Docker 镜像加速服务平台 编辑下面/etc/docker/daemon.json文件为 {"registry-mirrors": ["https://docker.xuanyuan.me"] }同时可以进入轩…...

是否存在路径(FIFOBB算法)

题目描述 一个具有 n 个顶点e条边的无向图&#xff0c;该图顶点的编号依次为0到n-1且不存在顶点与自身相连的边。请使用FIFOBB算法编写程序&#xff0c;确定是否存在从顶点 source到顶点 destination的路径。 输入 第一行两个整数&#xff0c;分别表示n 和 e 的值&#xff08;1…...

初探Service服务发现机制

1.Service简介 Service是将运行在一组Pod上的应用程序发布为网络服务的抽象方法。 主要功能&#xff1a;服务发现和负载均衡。 Service类型的包括ClusterIP类型、NodePort类型、LoadBalancer类型、ExternalName类型 2.Endpoints简介 Endpoints是一种Kubernetes资源&#xf…...

MySQL 索引底层结构揭秘:B-Tree 与 B+Tree 的区别与应用

文章目录 一、背景知识&#xff1a;什么是 B-Tree 和 BTree&#xff1f; B-Tree&#xff08;平衡多路查找树&#xff09; BTree&#xff08;B-Tree 的变种&#xff09; 二、结构对比&#xff1a;一张图看懂 三、为什么 MySQL InnoDB 选择 BTree&#xff1f; 1. 范围查询更快 2…...

实战三:开发网页端界面完成黑白视频转为彩色视频

​一、需求描述 设计一个简单的视频上色应用&#xff0c;用户可以通过网页界面上传黑白视频&#xff0c;系统会自动将其转换为彩色视频。整个过程对用户来说非常简单直观&#xff0c;不需要了解技术细节。 效果图 ​二、实现思路 总体思路&#xff1a; 用户通过Gradio界面上…...

消息队列系统设计与实践全解析

文章目录 &#x1f680; 消息队列系统设计与实践全解析&#x1f50d; 一、消息队列选型1.1 业务场景匹配矩阵1.2 吞吐量/延迟/可靠性权衡&#x1f4a1; 权衡决策框架 1.3 运维复杂度评估&#x1f527; 运维成本降低策略 &#x1f3d7;️ 二、典型架构设计2.1 分布式事务最终一致…...

机器学习的数学基础:线性模型

线性模型 线性模型的基本形式为&#xff1a; f ( x ) ω T x b f\left(\boldsymbol{x}\right)\boldsymbol{\omega}^\text{T}\boldsymbol{x}b f(x)ωTxb 回归问题 利用最小二乘法&#xff0c;得到 ω \boldsymbol{\omega} ω和 b b b的参数估计$ \boldsymbol{\hat{\omega}}…...

链式法则中 复合函数的推导路径 多变量“信息传递路径”

非常好&#xff0c;我们将之前关于偏导数链式法则中不能“约掉”偏导符号的问题&#xff0c;统一使用 二重复合函数&#xff1a; z f ( u ( x , y ) , v ( x , y ) ) \boxed{z f(u(x,y),\ v(x,y))} zf(u(x,y), v(x,y))​ 来全面说明。我们会展示其全微分形式&#xff08;偏导…...