从零开始解剖Spring Boot启动流程:一个Java小白的奇幻冒险之旅
大家好呀!今天我们要一起探索一个神奇的话题——Spring Boot的启动流程。我知道很多小伙伴一听到"启动流程"四个字就开始头疼,别担心!我会用最通俗易懂的方式,带你从main()
方法开始,一步步揭开Spring Boot的神秘面纱,直到内嵌Tomcat欢快地跑起来~ 🏃♂️
准备好了吗?让我们开始这段奇妙的旅程吧!🚀
第一章:一切的开始——main()方法
1.1 那个我们熟悉的入口
每个Spring Boot项目都有一个main()
方法,它就像是我们家的前门钥匙🔑,没有它,我们连门都进不去!
@SpringBootApplication
public class MyApplication {public static void main(String[] args) {SpringApplication.run(MyApplication.class, args);}
}
看到这个@SpringBootApplication
注解了吗?它其实是个"套娃"注解,里面包含了三个重要注解:
@SpringBootConfiguration
:标识这是个配置类@EnableAutoConfiguration
:开启自动配置魔法 ✨@ComponentScan
:告诉Spring去哪里找组件
1.2 SpringApplication.run() 做了什么?
当我们调用SpringApplication.run()
时,背后发生了很多事情:
- 创建SpringApplication实例:就像组装一台新电脑 💻
- 运行SpringApplication:按下开机键!
// 伪代码简化版
public static ConfigurableApplicationContext run(Class primarySource, String... args) {// 1. 创建SpringApplication实例SpringApplication application = new SpringApplication(primarySource);// 2. 运行!return application.run(args);
}
第二章:SpringApplication的构造过程 �
2.1 确定应用类型
Spring Boot会先判断我们是什么类型的应用:
- Web应用(Servlet):比如用Tomcat
- Reactive Web应用:比如用Netty
- 非Web应用:普通的Java应用
// 判断逻辑大致是这样的
private WebApplicationType deduceWebApplicationType() {if (ClassUtils.isPresent("org.springframework.web.reactive.DispatcherHandler", null) {return WebApplicationType.REACTIVE;}if (ClassUtils.isPresent("javax.servlet.Servlet", null)) {return WebApplicationType.SERVLET;}return WebApplicationType.NONE;
}
2.2 加载初始化器(Initializers)
初始化器就像是Spring Boot的"小助手"们,它们会在应用启动的不同阶段帮忙做一些准备工作。Spring Boot会从META-INF/spring.factories
文件中加载这些初始化器。
# 示例 spring.factories 内容
org.springframework.context.ApplicationContextInitializer=\
com.example.MyInitializer,\
com.example.AnotherInitializer
2.3 加载监听器(Listeners)
监听器就像是一群"小耳朵"👂,它们会监听Spring Boot启动过程中的各种事件,比如:
ApplicationStartingEvent
:应用刚开始启动ApplicationEnvironmentPreparedEvent
:环境准备好了ApplicationPreparedEvent
:应用上下文准备好了ApplicationStartedEvent
:应用启动完成ApplicationReadyEvent
:应用准备就绪
第三章:run()方法的奇幻旅程 🎢
现在,我们来到了最精彩的部分——run()
方法的执行过程!让我们一步步来看:
3.1 第一步:启动计时器 ⏱️
Spring Boot一启动就会打开秒表,记录启动耗时:
StopWatch stopWatch = new StopWatch();
stopWatch.start();
3.2 第二步:准备环境 🌍
Spring Boot会准备运行环境,这包括:
- 创建环境对象(根据应用类型不同创建不同的环境)
- 配置环境(读取配置文件,如application.properties/yml)
- 发布
ApplicationEnvironmentPreparedEvent
事件
ConfigurableEnvironment environment = prepareEnvironment(...);
3.3 第三步:创建应用上下文 🏗️
根据应用类型创建不同的应用上下文:
- Web应用:
AnnotationConfigServletWebServerApplicationContext
- Reactive Web应用:
AnnotationConfigReactiveWebServerApplicationContext
- 非Web应用:
AnnotationConfigApplicationContext
context = createApplicationContext();
3.4 第四步:准备上下文 �
这一步会做很多重要工作:
- 准备环境
- 后置处理Bean定义
- 应用所有初始化器
- 发布
ApplicationContextInitializedEvent
事件 - 注册Spring Bean
prepareContext(context, environment, listeners, applicationArguments, printedBanner);
3.5 第五步:刷新上下文 🔄
这是整个启动过程中最复杂的一步!refresh()
方法做了很多事情:
refreshContext(context);
让我们深入看看refresh()
方法都做了什么:
3.5.1 准备刷新
- 记录启动时间
- 初始化占位符解析器
- 验证必要的属性
3.5.2 获取BeanFactory
- 创建
DefaultListableBeanFactory
- 加载Bean定义
3.5.3 准备BeanFactory
- 设置类加载器
- 添加Bean后置处理器
- 注册环境Bean
3.5.4 执行BeanFactory后置处理器
- 执行
BeanFactoryPostProcessor
的实现类 - 这里会处理自动配置类(
@EnableAutoConfiguration
的核心)
3.5.5 注册Bean后置处理器
- 注册各种
BeanPostProcessor
- 这些处理器会在Bean创建前后执行一些逻辑
3.5.6 初始化消息源
- 国际化相关
3.5.7 初始化事件广播器
- 用于发布应用事件
3.5.8 初始化其他特殊Bean
- 比如Tomcat、Jetty等Web服务器
3.5.9 完成BeanFactory初始化
- 初始化所有剩余的单例Bean
3.5.10 完成刷新
- 发布
ContextRefreshedEvent
事件 - 启动Web服务器(比如Tomcat)
3.6 第六步:收尾工作 🎬
启动完成后:
- 停止计时器
- 发布
ApplicationStartedEvent
和ApplicationReadyEvent
事件 - 返回应用上下文
afterRefresh(context, applicationArguments);
stopWatch.stop();
第四章:内嵌Tomcat的启动奥秘 🐱
现在,让我们重点看看内嵌Tomcat是如何启动的!这是很多小伙伴最感兴趣的部分~
4.1 Tomcat在哪里被创建?
在refresh()
方法的"初始化其他特殊Bean"阶段,Spring Boot会创建Web服务器。对于Tomcat来说,关键类是这个:
org.springframework.boot.web.embedded.tomcat.TomcatServletWebServerFactory
4.2 Tomcat启动流程
-
创建Tomcat实例:
Tomcat tomcat = new Tomcat();
-
配置基础信息:
- 设置端口(默认8080)
- 设置上下文路径(默认"/")
- 配置连接器等
-
创建Web应用上下文:
Context context = tomcat.addContext("", docBase);
-
配置Servlet容器:
- 添加DispatcherServlet
- 配置过滤器等
-
启动Tomcat:
tomcat.start();
4.3 内嵌Tomcat vs 传统Tomcat
特点 | 内嵌Tomcat | 传统Tomcat |
---|---|---|
启动方式 | 通过Java代码启动 | 通过startup.sh/bat脚本启动 |
部署方式 | 打包成可执行JAR | 打包成WAR部署到webapps目录 |
配置方式 | 通过application.properties配置 | 通过server.xml配置 |
类加载器 | 使用应用的类加载器 | 使用Tomcat的类加载器 |
版本控制 | 由Spring Boot管理版本 | 需要单独安装和管理 |
4.4 为什么选择内嵌服务器?
- 简化部署:只需要一个JAR文件,不需要额外安装Tomcat
- 版本统一:Spring Boot管理Tomcat版本,避免冲突
- 快速启动:内嵌服务器启动更快
- 微服务友好:适合云原生和容器化部署
- 配置简单:通过配置文件即可完成大部分配置
第五章:自动配置的魔法 ✨
Spring Boot最强大的特性之一就是自动配置,让我们看看它是如何工作的!
5.1 @EnableAutoConfiguration 的秘密
这个注解会导入AutoConfigurationImportSelector
,它会从META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports
文件中加载自动配置类。
5.2 条件注解的妙用
自动配置类通常使用各种条件注解来决定是否生效:
@ConditionalOnClass
:当类路径下有指定类时生效@ConditionalOnMissingBean
:当容器中没有指定Bean时生效@ConditionalOnProperty
:当配置属性满足条件时生效
例如,Tomcat的自动配置类:
@Configuration(proxyBeanMethods = false)
@ConditionalOnClass({ Servlet.class, Tomcat.class, UpgradeProtocol.class })
@ConditionalOnMissingBean(value = ServletWebServerFactory.class, search = SearchStrategy.CURRENT)
public class EmbeddedTomcatAutoConfiguration {// 配置代码
}
5.3 自动配置的执行顺序
- 加载所有自动配置类
- 过滤掉不满足条件的配置类
- 按特定顺序应用剩余配置类
- 创建和配置各种Bean
第六章:Spring Boot启动流程图解 🗺️
为了帮助大家更好地理解,我画了一个简化的启动流程图:
main()│├─ 创建SpringApplication实例│ ├─ 确定应用类型│ ├─ 加载初始化器│ └─ 加载监听器│└─ 执行run()方法├─ 启动计时器├─ 准备环境├─ 创建应用上下文├─ 准备上下文├─ 刷新上下文 (核心!)│ ├─ 准备刷新│ ├─ 获取BeanFactory│ ├─ 准备BeanFactory│ ├─ 执行BeanFactory后置处理器│ ├─ 注册Bean后置处理器│ ├─ 初始化消息源│ ├─ 初始化事件广播器│ ├─ 初始化特殊Bean (如Tomcat)│ ├─ 完成BeanFactory初始化│ └─ 完成刷新├─ 收尾工作└─ 返回应用上下文
第七章:常见问题解答 ❓
7.1 为什么我的Spring Boot应用启动很慢?
可能原因:
- 类路径下JAR太多
- 自动配置类太多
- 数据库连接等初始化耗时
- 大量Bean需要初始化
解决方案:
- 使用
@Lazy
延迟初始化 - 排除不必要的自动配置
- 减少启动时扫描的包路径
7.2 如何自定义内嵌Tomcat配置?
在application.properties中:
server.port=9090
server.tomcat.max-threads=200
server.tomcat.connection-timeout=5000
或者通过代码:
@Bean
public WebServerFactoryCustomizer tomcatCustomizer() {return factory -> factory.addConnectorCustomizers(connector -> {connector.setPort(9090);connector.setProperty("maxThreads", "200");});
}
7.3 如何查看自动配置的过程?
启动时添加debug参数:
java -jar myapp.jar --debug
或者在application.properties中:
debug=true
第八章:启动优化小技巧 ⚡
8.1 组件扫描优化
默认情况下,Spring Boot会扫描主类所在包及其子包。如果项目很大,可以明确指定扫描路径:
@ComponentScan("com.myapp.package1", "com.myapp.package2")
8.2 延迟初始化
Spring Boot 2.2+支持全局延迟初始化:
spring.main.lazy-initialization=true
8.3 排除自动配置
如果知道某些自动配置不需要,可以排除它们:
@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class})
8.4 使用Spring Boot DevTools
在开发环境中,添加DevTools可以加快重启速度:
org.springframework.bootspring-boot-devtoolsruntime
第九章:总结与展望 🏁
哇!我们终于走完了Spring Boot的整个启动流程!让我们回顾一下重点:
- main()方法是入口,调用
SpringApplication.run()
- SpringApplication初始化时会确定应用类型、加载初始化和监听器
- run()方法是核心,完成了环境准备、上下文创建和刷新
- 刷新上下文是最复杂的部分,完成了Bean工厂准备、自动配置、Bean注册等
- 内嵌Tomcat在刷新阶段被创建和启动
- 自动配置通过条件注解智能地配置应用
Spring Boot的启动过程就像是一个精密的瑞士手表⌚,各个部件协同工作,最终让我们的应用顺利运行。理解这个过程不仅能帮助我们更好地使用Spring Boot,还能在遇到问题时快速定位原因。
希望这篇文章能帮助你理解Spring Boot的启动机制!如果觉得有用,别忘了点赞收藏哦~ 😊
下次见!👋
推荐阅读文章
-
由 Spring 静态注入引发的一个线上T0级别事故(真的以后得避坑)
-
如何理解 HTTP 是无状态的,以及它与 Cookie 和 Session 之间的联系
-
HTTP、HTTPS、Cookie 和 Session 之间的关系
-
什么是 Cookie?简单介绍与使用方法
-
什么是 Session?如何应用?
-
使用 Spring 框架构建 MVC 应用程序:初学者教程
-
有缺陷的 Java 代码:Java 开发人员最常犯的 10 大错误
-
如何理解应用 Java 多线程与并发编程?
-
把握Java泛型的艺术:协变、逆变与不可变性一网打尽
-
Java Spring 中常用的 @PostConstruct 注解使用总结
-
如何理解线程安全这个概念?
-
理解 Java 桥接方法
-
Spring 整合嵌入式 Tomcat 容器
-
Tomcat 如何加载 SpringMVC 组件
-
“在什么情况下类需要实现 Serializable,什么情况下又不需要(一)?”
-
“避免序列化灾难:掌握实现 Serializable 的真相!(二)”
-
如何自定义一个自己的 Spring Boot Starter 组件(从入门到实践)
-
解密 Redis:如何通过 IO 多路复用征服高并发挑战!
-
线程 vs 虚拟线程:深入理解及区别
-
深度解读 JDK 8、JDK 11、JDK 17 和 JDK 21 的区别
-
10大程序员提升代码优雅度的必杀技,瞬间让你成为团队宠儿!
-
“打破重复代码的魔咒:使用 Function 接口在 Java 8 中实现优雅重构!”
-
Java 中消除 If-else 技巧总结
-
线程池的核心参数配置(仅供参考)
-
【人工智能】聊聊Transformer,深度学习的一股清流(13)
-
Java 枚举的几个常用技巧,你可以试着用用
-
由 Spring 静态注入引发的一个线上T0级别事故(真的以后得避坑)
-
如何理解 HTTP 是无状态的,以及它与 Cookie 和 Session 之间的联系
-
HTTP、HTTPS、Cookie 和 Session 之间的关系
-
使用 Spring 框架构建 MVC 应用程序:初学者教程
-
有缺陷的 Java 代码:Java 开发人员最常犯的 10 大错误
-
Java Spring 中常用的 @PostConstruct 注解使用总结
-
线程 vs 虚拟线程:深入理解及区别
-
深度解读 JDK 8、JDK 11、JDK 17 和 JDK 21 的区别
-
10大程序员提升代码优雅度的必杀技,瞬间让你成为团队宠儿!
-
探索 Lombok 的 @Builder 和 @SuperBuilder:避坑指南(一)
-
为什么用了 @Builder 反而报错?深入理解 Lombok 的“暗坑”与解决方案(二)
相关文章:
从零开始解剖Spring Boot启动流程:一个Java小白的奇幻冒险之旅
大家好呀!今天我们要一起探索一个神奇的话题——Spring Boot的启动流程。我知道很多小伙伴一听到"启动流程"四个字就开始头疼,别担心!我会用最通俗易懂的方式,带你从main()方法开始,一步步揭开Spring Boot的…...
概率多假设跟踪(PMHT):多目标跟踪中的概率软关联与高效跟踪算法解析
一、PMHT 的起源与核心定位 (一)背景 在多目标跟踪中,传统算法面临以下瓶颈: JPDA:单帧局部最优关联,无法处理跨帧长时间断联,且假设目标数固定(如雷达跟踪中预设目标数范围&…...
4.信号和槽|存在意义|信号和槽的连接方式|信号和槽断开|lambda表达式|信号和槽优缺点(C++)
信号和槽存在意义 所谓的信号槽,终究要解决的问题,就是响应用户的操作 信号槽,其实在GUI开发的各种框架中,是一个比较有特色的存在 其他的GUI开发框架,搞的方式都要更简洁一些~~ 网页开发 (js dom api) 网…...
电脑 BIOS 操作指南(Computer BIOS Operation Guide)
电脑 BIOS 操作指南 电脑的BIOS界面(应为“BIOS”)是一个固件界面,允许用户配置电脑的硬件设置。 进入BIOS后,你可以进行多种设置,具体包括: 1.启动配置 启动顺序:设置从哪个设备启动&#x…...
Scrapeless Scraping Browser: A high-concurrency automation solution for AI
介绍:升级无缝抓取浏览器的并发能力 作为 Scrapeless 的开发者和创始团队,我们对人工智能自动化的未来充满真诚的热情。我们的使命是创建一个真正为 AI 设计的自动化浏览器。在过去的几年中,从 Browserless.io 到众多云服务供应商推出的“浏…...
Java项目—— 拼图小游戏(进阶版)
项目需求 在拼图小游戏基础版的基础上,完成下列要求: 一、实现更换拼图图片功能 1,给美女,动物,运动菜单按钮添加单击事件(动作监听) 2,当我们点击了美女之后,就会从13…...
解析:深度优先搜索、广度优先搜索和回溯搜索
一、深度优先搜索(DFS) 1. 原理 思想:从起始节点出发,顺着一条路径不断深入,直到到达目标或无路可走,然后回溯到最近的分支点,继续探索其他分支。 应用场景:路径查找、连通性检测、…...
探索大语言模型(LLM):循环神经网络的深度解析与实战(RNN、LSTM 与 GRU)
一、循环神经网络(RNN) 1.1 基本原理 循环神经网络之所以得名,是因为它在处理序列数据时,隐藏层的节点之间存在循环连接。这意味着网络能够记住之前时间步的信息,并利用这些信息来处理当前的输入。 想象一下…...
从零开始开发 MCP Server
作者:张星宇 在大型语言模型(LLM)生态快速演进的今天,Model Context Protocol(MCP)作为连接 AI 能力与真实世界的标准化协议,正逐步成为智能体开发的事实标准。该协议通过定义 Resources&#…...
Oracle日志系统之重做日志和归档日志
Oracle日志系统之重做日志和归档日志 重做日志归档日志 本文讨论Oracle日志系统中对数据恢复非常重要的两个日志:重做日志和归档日志。 重做日志 重做日志,英文名Redo Log,顾名思义,是用来数据重做的,主要使用场景是事…...
嵌入式开发--STM32G4系列硬件CRC支持MODBUS和CRC32
需求 在项目中,需要用到MODBUS CRC16校验,也要用到CRC32的校验,出于效率的考虑,准备用硬件CRC。 CRC 16的参数模型有很多种,我这里用的是MODBUS,对于不同的参数模型,会有不同的参数设置和初值&a…...
基于尚硅谷FreeRTOS视频笔记——4—多任务处理
目录 多任务处理 任务调度 任务的调度策略 优先级不同 优先级相同 多任务处理 通俗来讲就是 能够在同一时间 同时 进行多个任务的处理,这就时多任务处理。 但是,单核处理器一次只能处理一个任务,就是说在while中,任务们只能…...
中小型及初创企业如何实现数字化转型?
在当今动态的商业环境中,财务团队开始肩负起推动企业数字化转型的重任,即从传统的财务规划系统稳步迈向基于商业智能平台和以创新技术为驱动的解决方案领域。这些举措有望提高运营和分析效率,同时依托数据驱动的决策机制,帮助企业…...
java输出、输入语句
先创建一个用于测试的java 编写程序 #java.util使java标准库的一个包,这里拉取Scanner类 import java.util.Scanner;public class VariableTest {public static void main(String[] args) {#创建一个 Scanner 对象Scanner scanner new Scanner(System.in);System.…...
Python基础知识语法归纳总结(数据类型-1)
Python基础知识&语法归纳总结(数据类型) 一、Python基本数据类型 尤其注意,Python中的变量不需要特定的去声明,每个变量在使用前都必须对其进行赋值,它没有类型,我们所说的“类型”是变量所指的内存中对…...
Spring数据访问全解析:ORM整合与JDBC高效实践
目录 一、Spring ORM集成深度剖析 🌟 ORM模块架构设计 核心集成特性: 整合MyBatis示例配置: 二、Spring JDBC高效实践指南 🌟 传统JDBC vs Spring JDBC对比 🌟 JdbcTemplate核心操作示例 批量操作优化…...
哪种电脑更稳定?Mac?Windows?还是云电脑? 实测解密
随着科技的发展进步,电脑已成为当下各类群体的必备产品之一,它的妙用有很多,无论是学生党、打工人还是已经退休的人群或都离不开它的存在。然而,电脑虽好却也差异很大、不同品牌、不同系统、不同配置、不同价位的统统都会有区别。…...
【AI模型学习】关于写论文——论文的审美
文章目录 一、“补丁法”(Patching)1.1 介绍1.2 方法论1.3 实例 二、判断工作的价值2.1 介绍2.2 详细思路2.3 科研性vs工程性 三、novelty以及误区3.1 介绍3.2 举例 看了李沐老师的读论文系列后,总结三个老师提到的有关课题研究和论文写作的三…...
【面经】杭州产链数字科技一面
1.介绍一下自己 面试官您好!我叫***,目前是就读于****计算机科学与技术专业的一名学生。我平时在学校也自学了编程相关的知识,比如Java基础、Springboot、SpringCloud,关系型数据库Mysql,非关系型数据库Redisÿ…...
微信小程序调用yolo目标检测模型
目录 后端 前端微信小程序 完整代码 后端 利用Flask,调用目标检测模型,后端代码如下。 # flask_yolo.py from flask import Flask, request, jsonify from ultralytics import YOLO from PIL import Imageapp Flask(__name__) model_path best.p…...
vmware17 虚拟机 ubuntu22.04 桥接模式,虚拟机无法接收组播消息
问题描述: 在一个项目中,宿主机win10中,使用的vmware17pro 虚拟机安装的ubuntu22.04,按照网上的教程使用Qt绑定组播消息,在另外一个Ubuntu工控机上发送用wiresahrk抓包的组播消息 sudo tcpreplay -i enp1s0 --loop0 y…...
Kaggle-Bag of Words Meets Bags of Popcorn-(二分类+NLP+Bert模型)
Bag of Words Meets Bags of Popcorn 题意: 有很多条电影评论记录,问你每一条记录是积极性的评论还是消极性的评论。 数据处理: 1.首先这是文件是zip形式,要先解压,注意sep ‘\t’。 2.加载预训练的 BERT 分词器 …...
数字信号处理技术架构与功能演进
数字信号处理(DSP)是通过数字运算实现信号分析、变换、滤波及调制解调的技术领域,其发展过程与技术应用如下: 一、定义与核心功能 技术定义:通过算法将模拟信号转换为数字形式进行处理,具有高精度、可编程…...
IaaS架构剖析、场景实践
一、什么是 IaaS 1.1 定义 Infrastructure as a Service(IaaS,基础设施即服务)是一种按需、弹性提供计算、存储、网络和安全等底层 IT 资源的云服务模式。用户通过 API、CLI 或 Web 控制台即可在几分钟内创建、扩容或释放资源,而…...
国产之光DeepSeek架构理解与应用分析02
本专栏 国产之光DeepSeek架构理解与应用分析-CSDN博客 国产之光DeepSeek架构理解与应用分析02-CSDN博客 前置的一些内容理解 GPU TPU NPU的区别? 设计目的 GPU:最初是为了加速图形渲染而设计的,用于处理图像和视频数据,以提供高…...
EDID结构
EDID DDC通讯中传输显示设备数据 VGA , DVI 的EDID由128字节组成,hdmi的EDID增加扩展块128字节。扩展快的内容主要是和音频属性相关的,DVI和vga没有音频,hdmi自带音频,扩展快数据规范按照cea-861x标准。 Edid为了让pc或其他的图像…...
4.黑马学习笔记-SpringMVC(P43-P47)
1.SpringMVC简介 SpringMVC技术(更少的代码,简便)与servlet技术功能相同,属于web层开发技术。 SpringMVC是一种基于java实现MVC模型的轻量级web框架。 轻量级指的是(内存占用比较低,运行效率高)…...
CSS 文件格式
A QFrame#andrFrm[status"android_en"] A:表示父类或顶层窗口的类型。如果 A 是一个自定义的类名,确保该类已经正确注册到 Qt 系统中。QFrame:表示具体的控件类型。#andrFrm:表示控件的对象名称(通过 setOb…...
java输出HelloWorld
创建一个java格式文件,这里命令为HelloWorld 这里我选择用notepad编译,也可以直接用记事本 #public 访问修饰词,表示这个类可以被其他任何类访问 #class 定义类的关键字 #HelloWorld 类名,遵循驼峰命名法(首字母大写…...
【SAP ME 44】在 HANA DB中报废SFC时的SHOP_ORDER表记录锁定
症状 SELECT…FROM SHOP_ORDER FOR UPDATE 在 SFC 报废期间持有锁,当同时调用数量较大时,可能会导致 HANA 数据库出现大量锁积压。这有时会导致因等待 HANA 数据库释放“选择更新”锁而导致报废 SFC 花费数分钟。 HANA 数据库日志中的示例: # begin PreparedStatement_ex…...
《软件设计师》复习笔记(12.1)——范围管理、进度管理
目录 一、范围管理 1. 核心概念 2. 范围管理过程 WBS(工作分解结构)示例 真题示例: 二、进度管理 1. 核心过程 2. 关键工具与技术 真题示例: 一、范围管理 1. 核心概念 项目范围:为交付产品必须完成的工作…...
Git-使用教程(新手向)
一、基本概念: 1.Git,Github的关系: Git --- 本地用于管理代码的工具,可类比为游戏存档。(存档,仓库,项目在Git中是一个东西) Github --- 远程仓库平台,可类比为云端。…...
密码学中的盐值是什么?
目录 1. 盐值的基本概念 2. 盐值的作用 (1) 防止彩虹表攻击 (2) 防止相同的密码生成相同的哈希值 (3) 增加暴力破解的难度 3. 如何使用盐值? (1) 生成盐值 (2) 将盐值附加到密码 (3) 存储盐值和哈希值 (4) 验证密码 4. 盐值如何增加暴力破解的难度 在线暴…...
[工具]Java xml 转 Json
[工具]Java xml 转 Json 依赖 <!-- https://mvnrepository.com/artifact/cn.hutool/hutool-all --> <dependency><groupId>cn.hutool</groupId><artifactId>hutool-all</artifactId><version>5.8.37</version> </dependen…...
安全光幕的CE认证
在工业自动化飞速发展的当下,安全光幕作为保障操作人员安全的关键设备,其重要性不言而喻。对于想要进军欧盟市场的安全光幕制造商来说,CE 认证是必须跨越的一道关卡。今天,我们就来深入探讨安全光幕的 CE 认证流程。 什么是安全…...
DNS解析失败怎么解决?
在互联网时代,畅快地浏览网页、使用各类网络服务已成为生活常态。然而,当屏幕突然弹出 “DNS解析失败”的提示,原本顺畅的网络连接戛然而止,让人倍感困扰。DNS即域名系统,它如同互联网的 “电话簿”,负责将…...
亚马逊商品详情API数据接口概述,Amazon API
亚马逊商品详情API数据接口概述 亚马逊商品详情API(如Amazon Product Advertising API或Selling Partner API (SP-API))是亚马逊为开发者提供的官方接口,允许通过编程方式获取商品的详细信息,包括商品标题、价格、描述、图片、用…...
TCP/IP和UDP协议的发展历程
TCP/IP和UDP协议的发展历程 引言 互联网的发展史是人类技术创新的辉煌篇章,而在这一发展过程中,通信协议发挥了奠基性的作用。TCP/IP(传输控制协议/互联网协议)和UDP(用户数据报协议)作为互联网通信的基础…...
LeetCode 259 题全解析:Swift 快速找出“满足条件”的三人组
文章目录 摘要描述示例 1:示例 2:示例 3: 题解答案(Swift)题解代码分析示例测试及结果时间复杂度空间复杂度总结 摘要 本文围绕 LeetCode 259 题“较小的三数之和”,通过 Swift 给出两种解法,并…...
【MySQL】MySQL表的增删改查(CRUD) —— 上篇
目录 MySQL表的增删改查(CRUD) 1. 新增(Create)/插入数据 1.1 单行数据 全列插入 insert into 表名 values(值, 值......); 1.2 单行数据 指定列插入 1.3 多行数据 指定列插入 1.4 关于时间日期(datetime&am…...
基于大模型的腹股沟疝诊疗全流程风险预测与方案制定研究报告
目录 一、引言 1.1 研究背景与意义 1.2 国内外研究现状 1.3 研究目的与创新点 二、大模型技术概述 2.1 大模型基本原理 2.2 常用大模型类型及特点 2.3 大模型在医疗领域的应用潜力 三、腹股沟疝诊疗流程分析 3.1 腹股沟疝的发病机制与分类 3.2 传统术前评估方法与局…...
使用nssm将Nginx配置为Windows服务
使用nssm将Nginx配置为Windows服务 下载nssm工具 :使用NSSM创建服务启动并验证服务管理服务(启动/停止/重启) 下载nssm工具 : nssm下载网址 下载到指定路径下,解压就行。 使用NSSM创建服务 winr打开运行命令框&am…...
(8)VTK C++开发示例 --- 交互式3D部件
文章目录 1. 概述2. CMake链接VTK3. main.cpp文件4. 演示效果 更多精彩内容👉内容导航 👈👉VTK开发 👈 1. 概述 这个例子介绍了3D小部件(vtkBoxWidget)。3D小部件利用了前面介绍的事件/观察者设计模式。它们…...
ReAct、CoT 和 ToT:大模型提示词推理架构的对比分析
ReAct、CoT 和 ToT:大模型提示词推理架构的对比分析 在大型语言模型(LLM)的研究与应用中,如何有效提升模型在复杂任务上的推理能力是关键问题之一。目前,ReAct(Reasoning and Acting)、CoT&…...
Evidential Deep Learning和证据理论教材的区别(主要是概念)
最近终于彻底搞懂了Evidential Deep Learning,之前有很多看不是特别明白的地方,原来是和证据理论教材(是的,不只是国内老师写的,和国外的老师写的教材出入也比较大)的说法有很多不一样,所以特地…...
golang context源码
解析 context结构 Deadline:返回 context 的过期时间; Done:返回 context 中的 channel; Err:返回错误; Value:返回 context 中的对应 key 的值. type Context interface {Deadline() (deadl…...
VSCODE插值表达式失效问题
GET https://cdn.jsdelivr.net/npm/vue2.6.14/dist/vue.js net::ERR_CONNECTION_-CSDN博客 更换正确的vue域名 GET https://cdn.jsdelivr.net/npm/vue2.6.14/dist/vue.js net::ERR_CONNECTION_ <script src"https://unpkg.com/vue2.6.14/dist/vue.js"></sc…...
6.VTK 颜色
文章目录 概念RGB示例HSV示例 概念 RGB颜色系统:通过红(R)、绿(G)、蓝(B)三个颜色分量的组合来定义颜色。每个分量的取值范围是0到1,其中(0, 0, 0)代表黑色,而(1, 1, 1)代表白色。可以使用vtkProperty::SetColor(r, g, b)方法为Actor设置颜色…...
MQTTClient.c的线程模型与异步事件驱动
MQTTClient.c的线程模型与异步事件驱动 1. 多线程架构设计 MQTTClient.c通过分离网络I/O和用户逻辑线程实现异步通信,核心设计如下: sequenceDiagramparticipant 主线程 as 主线程(用户调用)participant 发送队列 as 发送队列pa…...
Flutter异常Couldn‘t find dynamic library in default locations
Flutter项目在Windows系统使用ffigen生成代码时报下面的错误: [SEVERE] : Couldnt find dynamic library in default locations. [SEVERE] : Please supply one or more path/to/llvm in ffigens config under the key llvm-path. Unhandled exception: Exception: …...