springboot如何启动内置tomcat?(源码详解)
SpringBoot项目启动时,源码无需依赖传统Tomcat,使用因为内部集成了Tomcat功能。源码本文将深入解析SpringBoot如何通过源码启动内置Tomcat。使用易版权源码
关键点在于`registerBeanPostProcessors`的源码`onRefresh`方法,它扩展了容器对象和bean实例化过程,使用确保单例和实例化完成。源码`initApplicationEventMuliticaster`则注册广播对象,使用与`applicationEvent`和`applicationListener`紧密相关。源码
文章的使用核心内容集中在`onRefresh()`方法,其中`createWenServer()`是源码关键。当`servletContext`和`webServer`为空时,使用会创建并初始化相关的源码组件,如`servletWebServerFactory`、`servletContext`(Web请求上下文)、`webServer`(抽象的web容器封装)和`WebServer`实例。`getWebServer()`方法允许在Spring容器刷新后连接webServer。
SpringBoot通过`TomcatServletWebServerFactory`获取webServer,该工厂负责创建和配置webServer,包括Tomcat组件的初始化,如`Connector`和`Context`的设置,以及与wrapper、engine、service和host等的关联。`new Connector`会根据传入的协议进行定制化配置。
理解了这些扩展点,QQip探针源码用户可以自定义配置,通过`ServerProperties`或自定义`tomcatConnectorCustomizers`和`tomcatProtocolHandlerCustomizers`来扩展Tomcat的连接器和协议处理器。这就是SpringBoot设计的巧妙之处。
最后,SpringBoot的启动流程涉及逐层初始化和启动Tomcat的组件,如engine、context和wrapper,它们通过生命周期方法如`init`、`start`和`destroy`协同工作。启动过程本质上是一个链式调用,每个组件的初始化和启动都会触发下一层组件的逻辑。
Springboot之分布式事务框架Seata实现原理源码分析
在Springboot 2.2. + Seata 1.3.0环境中,Seata通过GlobalTransactionScanner实现全局事务管理。首先,它会扫描带有@GlobalTransactional注解的方法类,作为BeanPostProcessor处理器,通过InstantiationAwareBeanPostProcessor的postProcessAfterInitialization方法中的wrapIfNecessary方法进行全局事务拦截。
GlobalTransactionScanner判断类方法是否有@GlobalTransactional注解,如果没有则直接返回,否则创建GlobalTransactionalInterceptor。拦截器负责全局事务的执行,包括事务开始、执行本地业务、提交和回滚等步骤。例如,事务开始时,Seata通过SPI技术将xid绑定到当前线程,神马asp源码执行过程中会记录undo log以实现回滚。
Seata自动配置会创建代理数据源(DataSourceProxy),在数据源方法调用时进行代理处理。当调用带有全局事务的方法时,如RestTemplate和Feign,拦截器会传递XID到请求头中,确保跨服务的事务一致性。参与者(被调用服务)通过SeataHandlerInterceptor拦截器获取并绑定XID,然后通过ConnectionProxy代理进行数据库操作,其中ConnectionContext用于判断是否为全局事务。
总结来说,Seata的核心机制是通过代理、拦截器和XID的传递,确保分布式环境下的事务处理协调和一致性。
SpringBoot 实战:优雅的使用枚举参数(原理篇)
探讨 Spring 如何优雅地处理枚举参数的原理。
深入 Spring 的核心,从请求入口 DispatcherServlet 开始,逐步解析参数处理逻辑。首先,通过 InvocableHandlerMethod#invokeForRequest 方法处理参数,调用 doInvoke 方法获取返回值。
在解析参数的逻辑中,找到了 RequestParamMethodArgumentResolver#resolveArgument 方法。在这里,不论目标参数为何类型,输入始终为 String 类型或 String 数组。Spring 则负责将这些基础类型转换为目标期望的iPhone开关源码类型。
在 DispatcherServlet 中,根据 URI 查找对应的 Controller 方法,从而确定目标参数类型。随后,通过 DataBinder#convertIfNecessary 检查是否需要进行转换,将字符串转换为目标枚举值。
进一步深入,发现转换器查找逻辑主要在于 TypeConverterDelegate#convertIfNecessary 方法。此方法通过检查 org.springframework.core.convert.support.GenericConversionService#canConvert 方法,判断转换是否可行。若可行,则执行具体的类型转换。
Spring 使用 Map 缓存通用转换器接口 GenericConverter,并通过 ConcurrentReferenceHashMap 管理并发情况。然而,getConverter 方法未采用同步逻辑,可能在并发请求时带来性能损耗,但对 web 请求而言,此损耗优于同步等待。
查找转换器的核心逻辑在于 GenericConversionService.Converters#find 方法,通过源类型和目标类型组合查找对应转换器。Spring 还会遍历类型家族族谱,确保全面覆盖转换需求。
当确定可以进行转换后,调用 org.springframework.core.convert.support.GenericConversionService#convert 方法执行转换操作。此过程涉及 GenericConverter 对象的获取与使用,提供了优化请求内多次调用相同转换逻辑的Allatori混淆源码思路,例如采用内存缓存。
最终,Spring 通过 ConversionUtils.invokeConverter 方法调用转换器的转换逻辑,完成输入参数到目标类型的转换。具体实现细节可在实战篇中查阅。
总结,Spring 对枚举参数的处理过程清晰明了,从请求解析到参数转换,再到目标类型执行,确保了代码的优雅性和效率。对于 GET 请求和传参式 POST 请求(Form 模式),枚举参数的转换逻辑一致。然而,对于 HTTP Body 方式,由于使用了不同的处理逻辑,实现 body 参数的类型转换需要采取额外的策略。未来将对此进行深入探讨。
头秃了,二十三张图带你从源码了解SpringBoot启动流程!
源码版本
作者使用的是Spring Boot的2.4.0版本。不同版本的Spring Boot可能存在差异,建议读者与作者保持一致,以确保源码的一致性。
从哪入手
Spring Boot源码的研究起点是主启动类,即标注着`@SpringBootApplication`注解并且包含`main()`方法的类。这是Spring Boot启动的核心。
源码如何切分
SpringApplication中的静态`run()`方法是一个复杂的流程,它分为两步:创建`SpringApplication`对象和执行`run()`方法。接下来将分别介绍这两部分。
如何创建`SpringApplication`
创建`SpringApplication`的过程本质上是一个对象的生成,通过调试追踪,最终调用的构造方法如图所示。创建过程主要涉及三个阶段,我们将逐一进行深入。
设置应用类型
创建过程中的重要步骤是确定应用类型,这将直接影响项目的性质,如Web应用或非Web应用。应用类型由WebApplicationType枚举类决定,加载特定类(如DispatcherServlet)来判断。
设置初始化器
初始化器(ApplicationContextInitializer)用于在IOC容器刷新之前进行初始化操作,例如ServletContextApplicationContextInitializer。获取初始化器的方式是从SpringApplication中的方法调用开始的,最终通过`#SpringFactoriesLoader.loadSpringFactories()`方法从类路径加载。
设置监听器
监听器(ApplicationListener)负责监听特定的事件(如IOC容器刷新或关闭)。在Spring Boot中,使用SpringApplicationEvent事件来扩展监听器概念,主要在启动过程中触发。获取监听器的方式与初始化器相同,从spring.factories文件中加载。
总结
SpringApplication的构建为`run()`方法的执行铺平了道路,关键步骤包括设置应用类型、初始化器和监听器。注意,初始化器和监听器需要在spring.factories文件中声明,才能在构建过程中加载,此时IOC容器尚未创建,即使注入到容器中也不会生效。
执行`run()`方法
在构建结束后,到了启动的阶段,`run()`方法将执行一系列操作,分为八个步骤进行详细解析。
步骤1:获取并启动运行过程监听器
SpringApplicationRunListener监听器用于监听应用程序的启动过程,通过调用方法从spring.factories文件中获取运行监听器实例,并执行特定事件的广播。
步骤2:环境构建
构建过程包括加载系统和自定义配置(如application.properties),并广播事件通知监听器。
步骤3:创建IOC容器
执行容器创建过程,根据应用类型选择容器类型,此步骤仅创建容器,未进行其他操作。
步骤4:IOC容器的前置处理
这一步是容器刷新前的准备工作,关键操作是将主启动类注入容器,为后续自动化配置奠定基础。
步骤5:调用初始化器
执行构建过程中设置的初始化器,加载自定义的初始化器实现。
步骤6:加载启动类,注入容器
将主启动类加载到IOC容器中,作为自动配置的入口。
步骤7:两次事件广播
这一步涉及两次事件广播,包括ApplicationContextInitializedEvent和ApplicationPreparedEvent。
步骤8:刷新容器
容器刷新由Spring框架完成,包括资源初始化、上下文广播器等。
步骤9:IOC容器的后置处理
这一步是容器刷新后的扩展操作,通常用于打印结束日志等。
步骤:发出结束执行的事件
使用EventPublishingRunListener广播ApplicationStartedEvent事件,允许在IOC容器中注入的监听器响应。
步骤:执行Runners
Spring Boot提供了两种Runner,即CommandLineRunner和ApplicationRunner,用于定制额外操作。
总结
Spring Boot启动流程相对简洁,通过八个步骤详细描述了从创建到执行的整个过程。理解run()方法的执行流程、事件、初始化器和监听器的执行时间点是关键。
SpringBoot整合Activiti工作流(附源码)
依赖: 在新建springBoot项目时勾选activiti,或在已建立的springBoot项目中添加以下依赖: 数据源和activiti配置: 在activiti的默认配置中,process-definition-location-prefix指定activiti流程描述文件的前缀,启动时,activiti将自动寻找此路径下的文件并部署。suffix为String数组,表示描述文件的默认后缀名。 springMVC配置: 配置静态资源和直接访问页面,采用thymeleaf依赖解析视图,主要采用异步方式获取数据,通过angularJS进行前端数据处理与展示。 使用activiti: 配置数据源和activiti后,启动项目,activiti服务组件自动加入到spring容器中。使用注入方法直接访问。在非自动配置的spring环境中,可通过指定bean的init-method配置activiti服务组件。 案例:请假流程示例: 1. 员工申请请假 设置请假信息,完成申请时传入参数。 2. 老板审批请假 (1) 查询审批任务 老板查看需审批的请假任务,设置VacTask对象用于页面展示。 (2) 完成审批 传入审批结果和任务ID。根据结果进行流程跳转。 3. 查询请假记录 在history表中查询已完成的请假记录,设置VO对象展示。 4. 前端展示与操作 (1) 审批列表与操作 展示审批列表及操作示例,完成一个springBoot与activiti6.0整合示例项目的说明与代码。 完整项目代码参考: 推荐阅读: 1. SpringBoot内容聚合 2. 设计模式内容聚合 3. Mybatis内容聚合 4. 多线程内容聚合springboot启动运行特定代码
在Spring Boot中,我们可以通过使用ApplicationRunner或者CommandLineRunner接口来实现在Spring Boot启动时运行特定代码。
一、背景与需求说明
在Spring Boot应用中,有时我们需要在应用启动时运行一些特定的代码,比如进行数据初始化、预加载缓存等操作。为了满足这种需求,Spring Boot提供了两个接口:ApplicationRunner和CommandLineRunner。这两个接口都提供了一个run方法,Spring Boot应用启动后会执行这两个接口中的run方法。
二、使用ApplicationRunner或CommandLineRunner接口
要实现启动时运行特定代码,我们需要实现ApplicationRunner或CommandLineRunner接口,并覆盖其run方法。这两个接口的使用方式类似,一般来说,如果我们的代码需要和应用程序的参数(也就是命令行参数)交互,那么我们可以使用CommandLineRunner。如果我们的代码不需要和应用程序的参数交互,那么我们可以使用ApplicationRunner。以下是一个使用ApplicationRunner的例子:
java
@Component
public class MyStartupRunner implements ApplicationRunner {
@Override
public void run(ApplicationArguments args) throws Exception {
// 在这里编写你的启动时需要执行的代码
System.out.println("This code will run when the Spring Boot starts.");
}
}
三、使用@PostConstruct注解
除了上面的方式,我们还可以使用@PostConstruct注解来实现启动时运行特定代码。@PostConstruct注解用于在依赖项注入完成后立即执行方法,因此也可以用来在Spring Boot启动时执行特定代码。以下是一个使用@PostConstruct的例子:
java
@Component
public class MyStartupBean {
@PostConstruct
public void init() {
// 在这里编写你的启动时需要执行的代码
System.out.println("This code will run when the Spring Boot starts.");
}
}
以上就是在Spring Boot中实现在启动时运行特定代码的几种常见方式。这几种方式各有特点,我们可以根据具体的需求选择适合的方式。需要注意的是,这些代码会在Spring Boot的启动阶段执行,因此应该避免执行太重的操作,以免影响应用的启动速度。
2025-01-04 09:28
2025-01-04 08:36
2025-01-04 08:24
2025-01-04 07:53
2025-01-04 07:50