Spring Boot运行原理及功能实现方式

简介: Spring Boot运行原理及功能实现方式

前言

快速入门程序编写完了,我们发现springBoot程序开发比spring程序编写起来容易的多。配置简洁,依赖关系简单,启动运行容易。那么结下了我们我们就要思考一下入门程序中的这些功能是怎么实现的。

接下来我们从以下几个方面研究:


SpringBoot的启动依赖

启动器starter有什么作用

启动引导类是怎么运行的

内置的tomcat服务器原理

pom.xml文件分析

我们应用配置第一个就是依赖,这个依赖的作用到底是什么我们仔细来分析一下。


项目中的pom.xml中继承了一个坐标

<parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>2.7.3</version>
    <relativePath/> <!-- lookup parent from repository -->
</parent>

注意:这里parent的坐标被工程继承了,相当于这是一个父类,我们创建的工程是一个子类,用到了父类的东西。


打开spring-boot-starter-parent之后,发现他又继承了一个坐标。

image.png


继续打开spring-boot-dependencies之后,发现该文件中主要定义了两组信息,分别是各种依赖的版本号和所有依赖的坐标信息,并对声明的版本号做了一个引用。我们打开发现这里有两千多行,所有能配置的版本基本都包含了。

image.png


由于Spring Boot工程使用到了maven的聚合工程,所以这里我们可以认为spring-boot-dependencies就是父工程,子工程就是我们自己的项目。当我们子工程中使用<parent>继承父类之后,所有的版本就都由父类决定了。

image.png


可以看到子工程当中我们没有声明版本号,是因为所有的版本都由父类决定,这样做的好处是什么:解决了版本冲突。不同模块、不同功能之间使用的版本是不一样的,因此spring boot就为我们将所有的版本统一化了。


启动器starter

SpringBoot官方给出了好多个starter的定义,方便我们使用,而且名称都是如下格式

命名规则: spring-boot-starter-技术名称


starter定义了使用某种技术时对于依赖的固定搭配格式,也是一种最佳解决方案,使用starter可以帮助开发者减少依赖配置


项目中的pom.xml定义了使用springMVC技术,但是并没有写SpringMVC的坐标,而是添加了一个名字中包含starter的依赖

<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-web</artifactId>
</dependency>

在spring-boot-starter-web中又定义了若干个具体依赖的坐标

<dependencies>
  <dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter</artifactId>
    <version>2.7.3</version>
    <scope>compile</scope>
  </dependency>
  <dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-json</artifactId>
    <version>2.7.3</version>
    <scope>compile</scope>
  </dependency>
  <dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-tomcat</artifactId>
    <version>2.7.3</version>
    <scope>compile</scope>
  </dependency>
  <dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-web</artifactId>
    <version>5.3.22</version>
    <scope>compile</scope>
  </dependency>
  <dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-webmvc</artifactId>
    <version>5.3.22</version>
    <scope>compile</scope>
  </dependency>
</dependencies>

然后你可以继续打开任意一个依赖,发现里面又会包含一些依赖。

我们可以发现,这个starter中又包含了若干个坐标,其实就是使用SpringMVC开发通常都会使用到Json,使用json又离不开这里面定义的这些坐标,看来还真是方便,SpringBoot把我们开发中使用的东西能用到的都给提前做好了。你仔细看完会发现,里面有一些你没用过的。的确会出现这种过量导入的可能性,没关系,可以通过maven中的排除依赖剔除掉一部分。不过你不管它也没事,大不了就是过量导入呗。


总结:

使用starter可以帮开发者快速配置依赖关系。以前写依赖3个坐标的,现在写导入一个就搞定了,就是加速依赖配置的。


启动引导类

目前程序运行的入口就是SpringBoot工程创建时自带的那个类了,带有main方法的那个类,运行这个类就可以启动springBoot工程的运行

@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}

Spring Boot本身就是为了加速Spring程序而开发的,而Spring程序运行的基础是需要创建自己的Spring容器对象,并将所有的对象管理在容器里面,这时候你可能会疑惑,Spring boot加速了Spring程序,那spring boot中有没有容器呢,或者说这个容器还存在吗?答案是存在,我们可以修改上面的代码来验证一下。

@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(Application.class, args);
        Application bean = context.getBean(Application.class);
        System.out.println(bean);
    }
}

这里的ConfigurableApplicationContext继承了ApplicationContext,说明他就是一个容器,拿到容器之后再通过getBean()来拿到bean,打印在控制台。


image.png


可以看到控制台打印出来了对应的地址,说明它就是c从容器中拿到的一个对象,说明spring boot中存在一个容器,而SpringApplication.run(Application.class, args);就是运行Spring Boot容器,


@SpringBootApplication注解

接下来我们看一下@SpringBootApplication注解


image.png


根据箭头打开注解可以看到有Spring Boot配置以及自动配置,自动配置包等等,因此当程序加了@SpringBootApplication注解,他就会扫描这个类所在包下和子包下面的所有类。并且他还是spring boot的配置类,所以spring的一些配置文件就不用写了,配置类自动搞定了。


总结一下:这个启动引导类,创建了一个容器并将该容器运行起来,做了一个配置类,里面有一些自动配置的东西,然后我们创建的对象就可以放在引导类的包以及子包中,就可以扫描到了。


内置的服务器

我们在创建spring boot项目时,会勾选需要的依赖,然后导入对应的starter

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>

内置的tomcat服务器来研究几个问题


这个服务器在什么位置定义的

这个服务器是怎么运行的

这个服务器如果想换怎么换

内嵌Tomcat定义位置

说到定义的位置,我们可以通过两种方式来查看:


第一种

image.png


第二种

image.png


tomcat运行原理

tomcat本身就是一个Java项目,那么可不可以将tomcat中的一些东西直接放回项目中,让他变成一个Java代码?


当然可以,这里spring boot将tomcat以jar包的形式放到spring容器中让他变成一种对象,然后运行这个对象,把我们的项目启动起来,这就是它能够启动并且运行的原理。


修改服务器

如果我们不想用他提供的版本能不能修改呢?可以,我们可以通过<exclusions>标签来排除提供的版本,或者内置服务器,然后在再重新添加依赖来达到修改服务器版本或者更换服务器的目的。

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
    <exclusions>
        <exclusion>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-tomcat</artifactId>
        </exclusion>
    </exclusions>
</dependency>

image.png

当添加<exclusions>标签之后,就把内置的tomcat的服务器排除了,这时候你在启动项目,在控制台就不会看到tomcat相关的信息了,并且网页也无法访问了。


添加服务器

这时候你再添加你想要的使用的服务器,只需添加对应的坐标即可

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-tomcat</artifactId>
</dependency>

image.png


同样在控制台就会看到服务器相关的信息了,并且网页也可以正常访问了。不过这样就显得有点多此一举了,好好的东西不用,非要自己搞这些,所以还是建议使用内置的服务器。


更换内嵌服务器

那我们是否可以换个服务器呢?必须的嘛。根据SpringBoot的工作机制,用什么技术,加入什么依赖就行了。SpringBoot提供了3款内置的服务器


tomcat(默认): apache出品,粉丝多,应用面广,负载了若干较重的组件

jetty:更轻量级,负载性能远不及tomcat

undertow:负载性能勉强跑赢tomcatI

想用哪个,加个坐标就OK。前提是把tomcat排除掉,因为tomcat是默认加载的。

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
    <exclusions>
        <exclusion>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-tomcat</artifactId>
        </exclusion>
    </exclusions>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-jetty</artifactId>
</dependency>

现在就已经成功替换了web服务器,它的核心思想就是加入对应的坐标就可以了。

目录
打赏
0
1
0
0
87
分享
相关文章
|
2月前
|
SpringBoot自动配置的原理是什么?
Spring Boot自动配置核心在于@EnableAutoConfiguration注解,它通过@Import导入配置选择器,加载META-INF/spring.factories中定义的自动配置类。这些类根据@Conditional系列注解判断是否生效。但Spring Boot 3.0后已弃用spring.factories,改用新格式的.imports文件进行配置。
707 0
基于Java+Springboot+Vue开发的鲜花商城管理系统源码+运行
基于Java+Springboot+Vue开发的鲜花商城管理系统(前后端分离),这是一项为大学生课程设计作业而开发的项目。该系统旨在帮助大学生学习并掌握Java编程技能,同时锻炼他们的项目设计与开发能力。通过学习基于Java的鲜花商城管理系统项目,大学生可以在实践中学习和提升自己的能力,为以后的职业发展打下坚实基础。技术学习共同进步
386 7
深度剖析【Spring】事务:万字详解,彻底掌握传播机制与事务原理
在Java开发中,Spring框架通过事务管理机制,帮我们轻松实现了这种“承诺”。它不仅封装了底层复杂的事务控制逻辑(比如手动开启、提交、回滚事务),还提供了灵活的配置方式,让开发者能专注于业务逻辑,而不用纠结于事务细节。
SpringBoot实现文件上传下载功能
本文介绍了如何使用SpringBoot实现文件上传与下载功能,涵盖配置和代码实现。包括Maven依赖配置(如`spring-boot-starter-web`和`spring-boot-starter-thymeleaf`)、前端HTML页面设计、WebConfig路径映射配置、YAML文件路径设置,以及核心的文件上传(通过`MultipartFile`处理)和下载(利用`ResponseEntity`返回文件流)功能的Java代码实现。文章由Colorful_WP撰写,内容详实,适合开发者学习参考。
458 0
Spring 框架核心原理与实践解析
本文详解 Spring 框架核心知识,包括 IOC(容器管理对象)与 DI(容器注入依赖),以及通过注解(如 @Service、@Autowired)声明 Bean 和注入依赖的方式。阐述了 Bean 的线程安全(默认单例可能有安全问题,需业务避免共享状态或设为 prototype)、作用域(@Scope 注解,常用 singleton、prototype 等)及完整生命周期(实例化、依赖注入、初始化、销毁等步骤)。 解析了循环依赖的解决机制(三级缓存)、AOP 的概念(公共逻辑抽为切面)、底层动态代理(JDK 与 Cglib 的区别)及项目应用(如日志记录)。介绍了事务的实现(基于 AOP
基于Spring Data Redis与RabbitMQ实现字符串缓存和计数功能(数据同步)
总的来说,借助Spring Data Redis和RabbitMQ,我们可以轻松实现字符串缓存和计数的功能。而关键的部分不过是一些"厨房的套路",一旦你掌握了这些套路,那么你就像厨师一样可以准备出一道道饕餮美食了。通过这种方式促进数据处理效率无疑将大大提高我们的生产力。
168 32
Spring核心原理剖析与解说
每个部分都是将一种巨大并且复杂的技术理念传达为更易于使用的接口,而这就是Spring的价值所在,它能让你专注于开发你的应用,而不必从头开始设计每一部分。
162 32
Spring Boot 功能模块全解析:构建现代Java应用的技术图谱
Spring Boot不是一个单一的工具,而是一个由众多功能模块组成的生态系统。这些模块可以根据应用需求灵活组合,构建从简单的REST API到复杂的微服务系统,再到现代的AI驱动应用。
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问