Java从入门到天黑|06高质量男性SpringBoot入门及原理(基础总结版,强烈建议收藏)

Posted 孙叫兽

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java从入门到天黑|06高质量男性SpringBoot入门及原理(基础总结版,强烈建议收藏)相关的知识,希望对你有一定的参考价值。

目录

SpringBoot简介

Spring是为了解决企业级应用开发的复杂性而创建的,简化开发。

helloWorld

项目结构分析:

运行原理探究

以后我们导入依赖默认是不需要写版本;但是如果导入的包没有在依赖中管理着就需要手动配置版本了;

         主启动类

@EnableAutoConfiguration :开启自动配置功能

@AutoConfigurationPackage : 自动配置包

@Import({AutoConfigurationImportSelector.class}) :给容器导入组件 ;

结论:

SpringApplication.run分析

这个类主要做了以下四件事情:

字面量:普通的值 [ 数字,布尔值,字符串 ]

对象、Map(键值对)

数组( List、set )

修改SpringBoot的默认端口号

注入配置文件

yaml配置注入到实体类完全OK!

测试步骤:

结论:

多环境切换

SpringBoot会从这四个位置全部加载主配置文件;互补配置;

自动配置原理

一句话总结 : 根据当前不同的条件判断,决定这个配置类是否生效! 一但这个配置类生效;这个配置类就会给容器中添加各种组件;

xxxxAutoConfigurartion:自动配置类;给容器中添加组件xxxxProperties:封装配置文件中相关属性;

@Conditional派生注解(Spring注解版原生的@Conditional作用)

那么多的自动配置类,必须在一定的条件下才能生效;也就是说,我们加载了这么多的配置类,但不是 所有的都生效了。

我们可以通过启用   debug=true属性;来让控制台打印自动配置报告,这样我们就可以很方便的知道哪些自动配置类生效;

Negative matches:(没有启动,没有匹配成功的自动配置类:负匹配) Unconditional classes: (没有条件的类)

提高:自定义starter

命名规范:


1	#配置项目的访问路径
2	server.servlet.context-path=/sun

SpringBoot简介

Spring是一个开源框架,2003 年兴起的一个轻量级的Java 开发框架,作者:Rod Johnson  

Spring是为了解决企业级应用开发的复杂性而创建的,简化开发。

Spring是如何简化Java开发的?

为了降低Java开发的复杂性,Spring采用了以下4种关键策略:

1、基于POJO的轻量级和最小侵入性编程,所有东西都是bean

2、通过IOC,依赖注入(DI)和面向接口实现松耦合;

3、基于切面(AOP)和惯例进行声明式编程;

4、通过切面和模版减少样式代码,RedisTemplatexxxTemplate

什么是SpringBoot?

学过javaweb的同学就知道,开发一个web应用,从最初开始接触Servlet结合Tomcat, 跑出一个Hello Wolrld程序,是要经历特别多的步骤; 后来就用了框架Struts,再后来是SpringMVC,到了现在的SpringBoot,过一两年又会有其他web框架出现;你们有经历过框架不断的演进,然后自己开发项目所 有的技术也再不断的变化、改造吗?建议都可以去经历一遍;

言归正传,什么是SpringBoot呢,就是一个javaweb的开发框架,和SpringMVC类似,对比其他javaweb框架的好处,官方说是简化开发,约定大于配置, you can "just run",能迅速的开发web应用,几行代码开发一个http接口。

所有的技术框架的发展似乎都遵循了一条主线规律:从一个复杂应用场景 衍生 一种规范框架,人们只需要进行各种配置而不需要自己去实现它,这时候强大的配置功能成了优点;发展到一定程度之后,人们 根据实际生产应用情况,选取其中实用功能和设计精华,重构出一些轻量级的框架;之后为了提高开发 效率,嫌弃原先的各类配置过于麻烦,于是开始提倡约定大于配置,进而衍生出一些一站式的解决方   案。

是的这就是Java企业级应用->J2EE->spring->springboot的过程。

随着 Spring 不断的发展,涉及的领域越来越多,项目整合开发需要配合各种各样的文件,慢慢变得不那么易用简单,违背了最初的理念,甚至人称配置地狱。Spring   Boot   正是在这样的一个背景下被抽象出来的开发框架,目的为了让大家更容易的使用 Spring 、更容易的集成各种常用的中间件、开源软件;

Spring Boot 基于 Spring 开发,Spirng Boot 本身并不提供 Spring 框架的核心特性以及扩展功能,只是用于快速、敏捷地开发新一代基于 Spring 框架的应用程序。也就是说,它并不是用来替代 Spring 的解决方案,而是和 Spring 框架紧密结合用于提升 Spring 开发者体验的工具。Spring Boot 约定大于配置的核心思想,默认帮我们进行了很多设置,多数 Spring Boot  应用只需要很少的 Spring  配置。同时它集成了大量常用的第三方库配置(例如 RedisMongoDBJpaRabbitMQQuartz 等等),Spring Boot 应用中这些第三方库几乎可以零配置的开箱即用。

学过javaweb的同学就知道,开发一个web应用,从最初开始接触Servlet结合Tomcat, 跑出一个Hello Wolrld程序,是要经历特别多的步骤; 后来就用了框架Struts,再后来是SpringMVC,到了现在的SpringBoot,过一两年又会有其他web框架出现;你们有经历过框架不断的演进,然后自己开发项目所 有的技术也再不断的变化、改造吗?建议都可以去经历一遍;

言归正传,什么是SpringBoot呢,就是一个javaweb的开发框架,和SpringMVC类似,对比其他javaweb框架的好处,官方说是简化开发,约定大于配置, you can "just run",能迅速的开发web应用,几行代码开发一个http接口。

所有的技术框架的发展似乎都遵循了一条主线规律:从一个复杂应用场景 衍生 一种规范框架,人们只需要进行各种配置而不需要自己去实现它,这时候强大的配置功能成了优点;发展到一定程度之后,人们 根据实际生产应用情况,选取其中实用功能和设计精华,重构出一些轻量级的框架;之后为了提高开发 效率,嫌弃原先的各类配置过于麻烦,于是开始提倡约定大于配置,进而衍生出一些一站式的解决方   案。

是的这就是Java企业级应用->J2EE->spring->springboot的过程。

随着 Spring 不断的发展,涉及的领域越来越多,项目整合开发需要配合各种各样的文件,慢慢变得不那么易用简单,违背了最初的理念,甚至人称配置地狱。Spring   Boot   正是在这样的一个背景下被抽象出来的开发框架,目的为了让大家更容易的使用 Spring 、更容易的集成各种常用的中间件、开源软件;

Spring Boot 基于 Spring 开发,Spirng Boot 本身并不提供 Spring 框架的核心特性以及扩展功能,只是用于快速、敏捷地开发新一代基于 Spring 框架的应用程序。也就是说,它并不是用来替代 Spring 的解决方案,而是和 Spring 框架紧密结合用于提升 Spring 开发者体验的工具。Spring Boot 约定大于配置的核心思想,默认帮我们进行了很多设置,多数 Spring Boot  应用只需要很少的 Spring  配置。同时它集成了大量常用的第三方库配置(例如 RedisMongoDBJpaRabbitMQQuartz 等等),Spring Boot 应用中这些第三方库几乎可以零配置的开箱即用。

Spring Boot的主要优点:

为所有Spring开发者更快的入门

开箱即用,提供各种默认配置来简化项目配置

内嵌式容器简化Web项目

没有冗余代码生成和XML配置的要求

使用 Spring Boot 到底有多爽。晚上试试就知道!

helloWorld

准备工作

我们将学习如何快速的创建一个Spring Boot应用,并且实现一个简单的Http请求处理。通过这个例子对Spring Boot有一个初步的了解,并体验其结构简单、开发快速的特性。

我的环境准备:

java version "1.8.0_181" Maven-3.6.1 SpringBoot 2.x 最新版

开发工具: IDEA

创建基础项目说明:

Spring官方提供了非常方便的工具让我们快速构建应用 ,   Spring   Initializrhttps://start.spring.io/

项目创建方式一:使用Spring Initializr Web页面创建项目1、打开 https://start.spring.io/

2、填写项目信息

3、点击”Generate   Project“按钮生成项目;下载此项目

  4、解压项目包,并用IDEAMaven项目导入,一路下一步即可,直到项目导入完毕。

5、如果是第一次使用,可能速度会比较慢,包比较多、需要耐心等待一切就绪。

项目创建方式二:使用 IDEA 直接创建项目

1、创建一个新项目

2、选择spring initalizr , 可以看到默认就是去官网的快速构建工具那里实现

3、填写项目信息

4、选择初始化的组件(初学勾选 Web 即可)

5、填写项目路径

6、等待项目构建成功

项目结构分析:

通过上面步骤完成了基础项目的创建。就会自动生成以下文件。

1、程序的主启动类

2、一个 application.properties 配置文件

3、一个 测试类

4、一个 pom.xml

pom.xml 分析

1	<!-- 父依赖 -->
2	<parent>
3	<groupId>org.springframework.boot</groupId>
4	<artifactId>spring-boot-starter-parent</artifactId>
5	<version>2.2.5.RELEASE</version>
6	<relativePath/>
7	</parent>
8	
9	<dependencies>
10	<!-- web场景启动器 -->
11	<dependency>
12	<groupId>org.springframework.boot</groupId>
13	<artifactId>spring-boot-starter-web</artifactId>
14	</dependency>
15	<!-- springboot单元测试 -->
16	<dependency>
17	<groupId>org.springframework.boot</groupId>
18	<artifactId>spring-boot-starter-test</artifactId>
19	<scope>test</scope>
20	<!-- 剔除依赖 -->
21	<exclusions>
22	<exclusion>
23	<groupId>org.junit.vintage</groupId>
24	<artifactId>junit-vintage-engine</artifactId>
25	</exclusion>
26	</exclusions>
27	</dependency>
28	</dependencies>
29	
30	<build>
31	<plugins>
32	<!-- 打包插件 -->
33	<plugin>
34	<groupId>org.springframework.boot</groupId>
35	<artifactId>spring-boot-maven-plugin</artifactId>
36	</plugin>
37	</plugins>
38	</build>

编写HTTP接口

1、在主程序的同级目录下,新建一个controller包,一定要在同级目录下,否则识别不到

2、在包中新建一个HelloController

1	@RestController
2	public class HelloController {
3	
4	@RequestMapping("/hello")
5	public String hello() {
6	return "Hello World";
7	}
8	
9	}

3、编写完毕后,从主程序启动项目,浏览器发起请求,看页面返回;控制台输出了 Tomcat 访问的端口号

简单几步,就完成了一个web接口的开发,SpringBoot就是这么简单。所以我们常用它来建立我们的微服务项目!

 将项目打成jar包,点击 mavenpackage

如果遇到以上错误,可以配置打包时   跳过项目运行测试用例

1	<!--
2	在工作中,很多情况下我们打包是不想执行测试用例的
3	可能是测试用例不完事,或是测试用例会影响数据库数据
4	跳过测试用例执
5	-->
6	<plugin>
7	<groupId>org.apache.maven.plugins</groupId>
8	<artifactId>maven-surefire-plugin</artifactId>
9	<configuration>
10	<!--跳过项目运行测试用例-->
11	<skipTests>true</skipTests>
12	</configuration>
13	</plugin>

如果打包成功,则会在target目录下生成一个 jar

打成了jar包后,就可以在任何地方运行了!OK 

如何更改启动时显示的字符拼成的字母,SpringBoot呢? 也就是 banner 图案; 只需一步:到项目下的 resources 目录下新建一个banner.txt 即可。

图案可以到:https://www.bootschool.net/ascii 这个网站生成,然后拷贝到文件中即可!

SpringBoot这么简单的东西背后一定有故事,我们之后去进行一波源码分析!

运行原理探究

我们之前写的HelloSpringBoot,到底是怎么运行的呢,Maven项目,我们一般从pom.xml文件探究起;

     Pom.xml

父依赖   

其中它主要是依赖一个父项目,主要是管理项目的资源过滤及插件!

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

 点进去,发现还有一个父依赖

<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>2.2.5.RELEASE</version>
<relativePath>../../spring-boot-dependencies</relativePath>
</parent>

这里才是真正管理SpringBoot应用里面所有依赖版本的地方,SpringBoot的版本控制中心;

以后我们导入依赖默认是不需要写版本;但是如果导入的包没有在依赖中管理着就需要手动配置版本了;

启动器  spring-boot-starter

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

springboot-boot-starter-xxx:就是spring-boot的场景启动器

spring-boot-starter-web:帮我们导入了web模块正常运行所依赖的组件;

SpringBoot将所有的功能场景都抽取出来,做成一个个的starter (启动器),只需要在项目中引入这些starter即可,所有相关的依赖都会导入进来 , 我们要用什么功能就导入什么样的场景启动器即可 ;我们未来也可以自己自定义 starter

主启动类

分析完了 pom.xml 来看看这个启动类

默认的主启动类

1	//@SpringBootApplication 来标注一个主程序类 , 说明这是一个Spring Boot应用
2	@SpringBootApplication
3	public class SpringbootApplication {
4	
5	public static void main(String[] args) {
6	//以为是启动了一个方法,没想到启动了一个服务
7	SpringApplication.run(SpringbootApplication.class, args);
8	}
9	
10	}

但是一个简单的启动类并不简单!我们来分析一下这些注解都干了什么

@SpringBootApplication

作用:标注在某个类上说明这个类是SpringBoot的主配置类 , SpringBoot就应该运行这个类的main方法来启动SpringBoot应用;

进入这个注解:可以看到上面还有很多其他注解!

@SpringBootConfiguration @EnableAutoConfiguration @ComponentScan(
excludeFilters = {@Filter( type = FilterType.CUSTOM,
classes = {TypeExcludeFilter.class}
), @Filter(
type = FilterType.CUSTOM,
classes = {AutoConfigurationExcludeFilter.class}
)}
)
public @interface SpringBootApplication {
// ......
}
v

@ComponentScan

这个注解在Spring中很重要   ,它对应XML配置中的元素。

作用:自动扫描并加载符合条件的组件或者bean   , 将这个bean定义加载到IOC容器中

@SpringBootConfiguration

作用:SpringBoot的配置类 ,标注在某个类上 , 表示这是一个SpringBoot的配置类; 我们继续进去这个注解查看

1	// 点进去得到下面的 @Component
2	@Configuration
3	public @interface SpringBootConfiguration {}
4	
5	@Component
6	public @interface Configuration {}

这里的 @Configuration,说明这是一个配置类 ,配置类就是对应Springxml   配置文件; 里面的 @Component 这就说明,启动类本身也是Spring中的一个组件而已,负责启动应用! 我们回到 SpringBootApplication 注解中继续看

@EnableAutoConfiguration

@EnableAutoConfiguration :开启自动配置功能

以前我们需要自己配置的东西,而现在SpringBoot可以自动帮我们配置    ;    @EnableAutoConfiguration

告诉SpringBoot开启自动配置功能,这样自动配置才能生效;

点进注解接续查看:

@AutoConfigurationPackage : 自动配置包

1	@Import({Registrar.class})
2	public @interface AutoConfigurationPackage {
3	}

@import Spring底层注解@import , 给容器中导入一个组件

Registrar.class 作用:将主启动类的所在包及包下面所有子包里面的所有组件扫描到Spring容器 ; 这个分析完了,退到上一步,继续看

@Import({AutoConfigurationImportSelector.class}) :给容器导入组件 ;

AutoConfigurationImportSelector : 自动配置导入选择器,那么它会导入哪些组件的选择器呢? 我们点击去这个类看源码:

1、这个类中有一个这样的方法

// 获得候选的配置
protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {
//这里的getSpringFactoriesLoaderFactoryClass()方法
//返回的就是我们最开始看的启动自动导入配置文件的注解类;EnableAutoConfiguration List<String> configurations =
SpringFactoriesLoader.loadFactoryNames(this.getSpringFactoriesLoaderFactoryCl ass(), this.getBeanClassLoader());
Assert.notEmpty(configurations, "No auto configuration classes found in META-INF/spring.factories. If you are using a custom packaging, make sure that file is correct.");
return configurations;
}

2、这个方法又调用了    SpringFactoriesLoader 类的静态方法!我们进入SpringFactoriesLoader类loadFactoryNames() 方法

public static List<String> loadFactoryNames(Class<?> factoryClass, @Nullable ClassLoader classLoader) {
String factoryClassName = factoryClass.getName();
//这里它又调用了 loadSpringFactories 方法
return (List)loadSpringFactories(classLoader).getOrDefault(factoryClassName, Collections.emptyList());
}

3、我们继续点击查看 loadSpringFactories 方法

1	private static Map<String, List<String>> loadSpringFactories(@Nullable
	ClassLoader classLoader) {
2	//获得classLoader , 我们返回可以看到这里得到的就是EnableAutoConfiguration标注
	的类本身
3	MultiValueMap<String, String> result =
	(MultiValueMap)cache.get(classLoader);
4	if (result != null) {
5	return result;
6	} else {
7	try {
8	//去获取一个资源 "META-INF/spring.factories"
9	Enumeration<URL> urls = classLoader != null ?
	classLoader.getResources("META-INF/spring.factories") :
	ClassLoader.getSystemResources("META-INF/spring.factories");
10	LinkedMultiValueMap result = new LinkedMultiValueMap();
11	
12	//将读取到的资源遍历,封装成为一个Properties
13	while(urls.hasMoreElements()) {
14	URL url = (URL)urls.nextElement();
15	UrlResource resource = new UrlResource(url);
16	Properties properties =
	PropertiesLoaderUtils.loadProperties(resource);
17	Iterator var6 = properties.entrySet().iterator();
18	
19	while(var6.hasNext()) {
20	Entry<?, ?> entry = (Entry)var6.next();
21	String factoryClassName =
	((String)entry.getKey()).trim();
22	String[] var9 =
	StringUtils.commaDelimitedListToStringArray((String)entry.getValue());
23	int var10 = var9.length;
24	
25	for(int var11 = 0; var11 < var10; ++var11) {
26	String factoryName = var9[var11];
27	result.add(factoryClassName, factoryName.trim());
28	}
29	}
30	}
31	
32	cache.put(classLoader, result);
33	return result;
34	} catch (IOException var13) {
35	throw new IllegalArgumentException("Unable to load factories
	from location [META-INF/spring.factories]", var13);
36	}
37	}
38	}

4、发现一个多次出现的文件spring.factories,全局搜索

我们根据源头打开spring.factories   ,   看到了很多自动配置的文件;这就是自动配置根源所在!

WebMvcAutoConfiguration

我们在上面的自动配置类随便找一个打开看看,比如   :   WebMvcAutoConfiguration

可以看到这些一个个的都是JavaConfig配置类,而且都注入了一些Bean,可以找一些自己认识的类,看 着熟悉一下!

所以,自动配置真正实现是从classpath中搜寻所有的META-INF/spring.factories配置文件    ,并将其中对应的 org.springframework.boot.autoconfigure. 包下的配置项,通过反射实例化为对应标注了@ConfigurationJavaConfig形式的IOC容器配置类 , 然后将这些都汇总成为一个实例并加载到IOC容器中。

结论:

  1. SpringBoot在启动的时候从类路径下的META-INF/spring.factories中获取EnableAutoConfiguration指定的值
  2. 将这些值作为自动配置类导入容器 , 自动配置类就生效 , 帮我们进行自动配置工作;
  3. 整个J2EE的整体解决方案和自动配置都在springboot-autoconfigurejar包中;
  4. 它会给容器中导入非常多的自动配置类 xxxAutoConfiguration, 就是给容器中导入这个场景需要的所有组件 , 并配置好这些组件 ;
  5. 有了自动配置类 , 免去了我们手动编写配置注入功能组件等的工作;

现在大家应该大概的了解了下,SpringBoot的运行原理,后面我们还会深化一次!

SpringApplication

我最初以为就是运行了一个main方法,没想到却开启了一个服务;

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

 

SpringApplication.run分析

分析该方法主要分两部分,一部分是SpringApplication的实例化,二是run方法的执行

SpringApplication

这个类主要做了以下四件事情:

1、推断应用的类型是普通的项目还是Web项目

2、查找并加载所有可用初始化器 , 设置到initializers属性中

3、找出所有的应用程序监听器,设置到listeners属性中 4、推断并设置main方法的定义类,找到运行的主类

查看构造器:

1	public SpringApplication(ResourceLoader resourceLoader, Class...
	primarySources) {
2	// ......
3	this.webApplicationType = WebApplicationType.deduceFromClasspath();
4	this.setInitializers(this.getSpringFactoriesInstances();
5	
	this.setListeners(this.getSpringFactoriesInstances(ApplicationListener.class
	));
6	this.mainApplicationClass = this.deduceMainApplicationClass();
7	}

 run方法

Yaml语法学习

配置文件

SpringBoot使用一个全局的配置文件 , 配置文件名称是固定的

application.properties

语法结构 : key=value application.yml

以上是关于Java从入门到天黑|06高质量男性SpringBoot入门及原理(基础总结版,强烈建议收藏)的主要内容,如果未能解决你的问题,请参考以下文章

Java从入门到天黑|04JavaSE入门之数组

Java从入门到天黑|05JavaSE入门之面向对象(下)

Java从入门到天黑|05JavaSE入门之面向对象(下)

Java从入门到天黑|03JavaSE入门之流程控制

Java从入门到天黑|04JavaSE入门之数组

Java从入门到天黑|05JavaSE入门之面向对象(上)