SpringBoot为什么使用new RuntimeException() 来获取调用栈?

 为什么不直接使用 Thread.currentThread().getStackTrace()

这确实看起来有点“奇怪”或者“绕”,但其实这是 Java 中一种非常常见、巧妙且合法的技巧,用于在运行时动态获取当前代码的调用栈信息。

Spring 选择用 new RuntimeException().getStackTrace() 是有原因的,主要有以下几点区别:

特性new Exception().getStackTrace()Thread.currentThread().getStackTrace()
调用栈更详细✅ 包括每个类和方法名❌ 可能只包含类名,不包含具体方法名
性能开销略高(需要构造异常对象)较低
精确性✅ 更精确地定位到调用者❌ 在某些 JVM 实现中可能不准确
使用场景需要精确调用栈时(如框架内部)快速查看线程堆栈(调试、日志等)

所以 Spring 框架为了确保能够准确找到调用链中的 main 方法所在类,选择了第一种方式。

	public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {this.resourceLoader = resourceLoader;Assert.notNull(primarySources, "PrimarySources must not be null");// 注册启动类this.primarySources = new LinkedHashSet<>(Arrays.asList(primarySources));// 推断应用类型,一般是webthis.webApplicationType = WebApplicationType.deduceFromClasspath();// 注册初始化器 (扩展接口)setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));// 注册监听器  (扩展接口)setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));// 找到启动的主方法入口this.mainApplicationClass = deduceMainApplicationClass();}private Class<?> deduceMainApplicationClass() {try {// 使用RuntimeException().getStackTrace()获取调用栈StackTraceElement[] stackTrace = new RuntimeException().getStackTrace();for (StackTraceElement stackTraceElement : stackTrace) {if ("main".equals(stackTraceElement.getMethodName())) {return Class.forName(stackTraceElement.getClassName());}}}catch (ClassNotFoundException ex) {// Swallow and continue}return null;}
public enum WebApplicationType {/*** 不需要嵌入式 web 容器,不是 web 应用*/NONE,/*** 需要嵌入式的 web 容器(如 Tomcat, Jetty)*/SERVLET,/*** 使用 reactive web stack(如 Netty + WebFlux)*/REACTIVE;
}

补充:

public ConfigurableApplicationContext run(String... args) {StopWatch stopWatch = new StopWatch();stopWatch.start();ConfigurableApplicationContext context = null;configureHeadlessProperty();SpringApplicationRunListeners listeners = getRunListeners(args);// 监听器触发1: EventPublishingRunListener 会发布 ApplicationStartingEventlisteners.starting();try {ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);// 准备应用所需的环境信息ConfigurableEnvironment environment = prepareEnvironment(listeners, applicationArguments);configureIgnoreBeanInfo(environment);// 打印Banner的logo和日志Banner printedBanner = printBanner(environment);context = createApplicationContext();// 准备应用上下文,包括设置环境、注册初始 Bean 定义、触发上下文准备完成事件prepareContext(context, environment, listeners, applicationArguments, printedBanner);// 刷新应用上下文 (核心)refreshContext(context);// 应用上下文刷新后的一些操作afterRefresh(context, applicationArguments);stopWatch.stop();if (this.logStartupInfo) {new StartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(), stopWatch);}// 监听器触发2:ApplicationReadyListener发布 ApplicationReadyEventlisteners.started(context);// 调用所有 CommandLineRunner 和 ApplicationRunner 接口实现类的 run 方法callRunners(context, applicationArguments);}catch (Throwable ex) {handleRunFailure(context, ex, listeners);throw new IllegalStateException(ex);}try {// 监听器触发3:对于 running(context) 并没有直接对应的事件被发布,执行一些运行时的监控或通知逻辑,比如通知外部系统应用已就绪listeners.running(context);}catch (Throwable ex) {handleRunFailure(context, ex, null);throw new IllegalStateException(ex);}return context;}
  • starting() - 触发 ApplicationStartingEvent
  • environmentPrepared() - 触发 ApplicationEnvironmentPreparedEvent
  • contextPrepared() - 触发 ApplicationContextInitializedEvent
  • contextLoaded() - 触发 ApplicationPreparedEvent
  • started(context) - 触发 ApplicationReadyEvent

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

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

相关文章

小白成长之路-haproxy负载均衡

文章目录一、概述1、HAProxy简介2、HAProxy特点和优点&#xff1a;3、HAProxy保持会话的三种解决方法4、HAProxy的balance 8种负载均衡算法1&#xff09;RR&#xff08;Round Robin&#xff09;2&#xff09;LC&#xff08;Least Connections&#xff09;3&#xff09;SH&#…

Kafka 与 RocketMQ 消息确认机制对比分析

目录 生产者消息确认机制 Kafka 生产者 ACK 机制 RocketMQ 生产者确认机制 消费者消息确认机制 Kafka 消费者确认机制 RocketMQ 消费者确认机制 核心差异对比 选型建议 消息确认机制是分布式消息中间件的核心功能之一&#xff0c;它直接关系到消息传递的可靠性和系统性能…

C/C++---rdbuf()函数

在C中&#xff0c;rdbuf() 是I/O流库中的一个核心成员函数&#xff0c;主要用于访问和操作流对象的缓冲区。这个函数在底层数据处理、流重定向以及自定义流操作等场景中应用广泛。下面将从多个方面详细解析 rdbuf() 函数。 基本概念与函数原型 rdbuf() 是 std::basic_ios 类的成…

【LLM】从零到一构建一个小型LLM--MiniGPT

从零到一构建一个小型LLM (Small Language Model)暂时起名为MiniGPT。这个模型将专注于因果语言建模 (Causal Language Modeling)&#xff0c;这是许多现代LLM&#xff08;如GPT系列&#xff09;的核心预训练任务。模型设计&#xff1a; 我们设计的模型是一个仅包含解码器 (Dec…

网络安全威胁下的企业困境与破局技术实践

前言&#xff1a;网络安全威胁下的企业困境 在数字化转型的浪潮中&#xff0c;企业对信息技术的依赖程度日益加深&#xff0c;但随之而来的网络安全威胁也愈发严峻。据统计&#xff0c;全球每年因网络安全事件造成的经济损失高达数万亿美元&#xff0c;其中中小企业更是成为了网…

[RAG system] 信息检索器 | BM25 Vector | Pickle格式 | HybridRetriever重排序

第六章&#xff1a;信息检索器 在上一章中&#xff0c;我们成功完成了知识库摄入流程。这是巨大的进步~ 我们精心准备了文档"块"&#xff08;类似独立的索引卡&#xff09;&#xff0c;并将其存储在两套智能归档系统中&#xff1a;向量数据库&#xff08;用于基于含…

Android 高通平台修改音频参数效果文件-优化音频效果

Android 高通平台如何音频效果 修改音频参数效果文件-优化音频效果 按如下方式修改。 开发云 - 一站式云服务平台 diff --git a/vendor/qcom/proprietary/mm-audio/audcal/family-b/acdbdata//MTP/workspaceFile.qwsp b/vendor/qcom/proprietary/mm-audio/audcal/family-b/acdb…

Install Docker Engine on UbuntuMySQL

Install Docker Engine on Ubuntu&&MySQL安装docker安装mysql客户端连接数据库我真气鼠了&#xff0c;今天得到一个血泪的教训&#xff0c;以后一定看官方文档&#xff01;&#xff01;&#xff01;学的课用的centos&#xff0c;指令全是yum&#xff0c;我这边不通用&a…

智能人体感应模块HC-SR501应用指南---使用esp32

人体热释电探头红外感应模块 人体感应开关HC-SR501蓝板新款 绿板-淘宝网 HC-SR501 人体红外感应电子模块传感器热释电探头感应开关RD-624-tmall.com天猫 模块信息 HC-SR501人体感应开关是一种基于红外线技术的自动控制模块&#xff0c;广泛应用于安防、智能家居和自动控制等领…

加速度传感器方向校准方法

保持平板平放在桌面上&#xff0c;将后置摄像头保持在平板的左上后方&#xff0c;或者右上后方&#xff0c;此为机器的正方向 1、以一台重力方向正常的机器做测试&#xff0c;通过DeviceInfoHw这个软件的加速度测试功能【Accelerometer Test】我们可以知道 X方向数据测试&#…

【OpenHarmonyOS应用开发】

OpenHarmonyOS应用开发1.OpenHarmonyOS应用开发环境安装2.初始化项目3.连接润和软件的开发板套件1.OpenHarmonyOS应用开发环境安装 进入HarmonyOS下载鸿蒙应用开发工具DevEco Studio 5.0.7.200版本。 双击打开下载好的可执行文件&#xff0c;点击下一步。 如果已经安装过&am…

50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | AutoTextEffect(自动打字机)

&#x1f4c5; 我们继续 50 个小项目挑战&#xff01;—— AutoTextEffect组件 仓库地址&#xff1a;https://github.com/SunACong/50-vue-projects 项目预览地址&#xff1a;https://50-vue-projects.vercel.app/。 利用 Vue 3 的 Composition API 和一些简单的 CSS 动画来构…

[RAG] LLM 交互层 | 适配器模式 | 文档解析器(`docling`库, CNN, OCR, OpenCV)

第二章&#xff1a;LLM 交互层 在上一章中&#xff0c;我们学习了作为"项目总控"的管道协调器&#xff0c;它负责协调 RAG 系统中各个功能模块。 其中最重要的协调对象之一&#xff0c;便是负责与大型语言模型&#xff08;LLM&#xff09;进行智能交互的LLM 交互层…

Golang 并发快速上手

文章目录1. 为什么要用协程&#xff1f;1.1 进程与线程1.2 协程1.3 线程和协程的区别线程协程1.4 Go 协程&#xff08;goroutines&#xff09;和协程&#xff08;coroutines&#xff09;2.Go 协程基本内容2.1 channel2.2 select2.3 future 模式3. 实践示例3.1 并发处理多个网络…

ESP32轻松实现UDP无线通信

ESP32支持UDP通信&#xff0c;这是一种轻量级、高效的通信协议&#xff0c;适用于需要快速数据传输但对数据可靠性要求不高的场景。以下是关于ESP32如何实现UDP通信的详细说明&#xff1a; 1. UDP协议简介及其适用场景 UDP&#xff08;用户数据报协议&#xff09;是一种无连接的…

Electron实现“仅首次运行时创建SQLite数据库”

在桌面应用中&#xff0c;SQLite因其轻量、嵌入式特性成为本地存储的热门选择。但若重复初始化数据库&#xff0c;会导致数据覆盖或冗余。本文将详解如何让Electron应用仅在首次启动时创建SQLite数据库&#xff0c;后续启动直接连接现有库。一、核心逻辑与实现原理 核心思路&am…

阿里开源AI大模型ThinkSound如何为视频配上灵魂之声

目录 前言 一、当AI解决视频配音的困境 二、引入“思维链”&#xff1a;让AI像专业音效师一样思考 三、背后的技术支撑 四、未来ThinkSound会如何改变我们的世界&#xff1f; 总结 &#x1f3ac; 攻城狮7号&#xff1a;个人主页 &#x1f525; 个人专栏:《AI前沿技术要闻…

图论(1):多叉树

多叉树一、基础知识1. 图 & 树2. 模板2.1 建图二、简单循环1. 【模板】树的路径求和2. 道路修建&#xff08;改&#xff09;3. 联合权值4. 毛毛虫树三、自顶向下/自底向上1. 医疗中心2. 【模板】树的直径3. 【模板】最大子树和4. 信号放大器一、基础知识 1. 图 & 树 …

楼宇自动化:Modbus 在暖通空调(HVAC)中的节能控制(一)

引言**在当今的建筑领域&#xff0c;楼宇自动化正扮演着愈发关键的角色&#xff0c;它致力于提升建筑的舒适度、安全性以及能源效率。而暖通空调&#xff08;HVAC&#xff09;系统作为楼宇自动化中的核心部分&#xff0c;其能耗在整个建筑能耗中占比相当高&#xff0c;据相关数…

【SpringBoot】注册条件+自动配置原理+自定义starter

注册条件注入到容器内实体类型对象的属性都是null&#xff0c;这些对象并没有什么实际的意义&#xff0c;因为实体类的对象就是来封装对象的&#xff0c;结果你这些对象中什么都没有&#xff1b;解决方法是1.给这些属性赋值然后再注入bean但是这些属性又是固定的不是很好&#…