Spring Boot入门,源码解析
1、Spring Boot简介
简化Spring应用开发的一个框架
整个Spring技术栈的一个大整合
J2EE开发的一站式解决方案
2、微服务
2014,Martin Fowler微服务论文
微服务:架构风格(服务微化)
每一个功能元素最终都是一个可独立替换和独立升级的软件单元
一个应用应该是一组小型服务:可以通过HTTP的方式进行互通
单体应用
环境约束(我的开发环境):
- JDK8:java version "1.8.0_131"
- Maven3.x:Apache Maven 3.6.3
- IntelliJ IDEA2019.2
- SpringBoot 1.5.9.RELEASE
3、Spring Boot HelloWorld
一个功能:
浏览器发送请求,服务器接收请求并处理,响应"Hello World"字符串
3.1 创建一个Maven工程
3.2 导入依赖Spring Boot相关的依赖
<!-- Inherit defaults from Spring Boot -->
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.5.9.RELEASE</version>
</parent>
<!-- Add typical dependencies for a web application -->
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
</dependencies>
3.3 编写一个主程序
启动Spring Boot应用
@SpringBootApplication
@RestController
public class DemoApplication {
public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}
@GetMapping("/hello")
public String hello(@RequestParam(value = "name", defaultValue = "World") String name) {
return String.format("Hello %s!", name);
}
}
3.4 编写相关的Controller、Service层
3.5 运行主程序测试
3.6 简化部署
<!-- 这个插件可以将应用打包成一个可执行的jar包 -->
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
要运行该应用程序,请使用java -jar命令:
$ java -jar target/myproject-0.0.1-SNAPSHOT.jar
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v1.5.9.RELEASE)
....... . . .
....... . . . (log output here)
....... . . .
........ Started Example in 2.536 seconds (JVM running for 2.864)
4、Hello World探究
4.1 POM文件
父项目
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.5.9.RELEASE</version>
</parent>
Ctrl+鼠标左键查看底层源码,上面这个父项目还依赖于
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-dependencies</artifactId>
<version>1.5.9.RELEASE</version>
<relativePath>../../spring-boot-dependencies</relativePath>
</parent>
实现真正管理Spring Boot应用里面的所有依赖版本
Spring Boot的版本仲裁中心
<properties>
<!-- Dependency versions -->
<activemq.version>5.14.5</activemq.version>
<antlr2.version>2.7.7</antlr2.version>
...
<maven-war-plugin.version>2.6</maven-war-plugin.version>
<versions-maven-plugin.version>2.2</versions-maven-plugin.version>
</properties>
以后我们导入依赖默认是不需要写版本(没有在dependencies里面管理的依赖自然需要声明版本号)
4.2 导入的依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
spring-boot-starter-web
- spring-boot-starter:spring-boot场景启动器;帮我们导入了web模块正常运行所依赖的组件
- 更多启动器
- Spring Boot将所有的功能场景都抽取出来,做成一个个的starter(启动器),只需要在项目里引入这些starter相关场景的所有依赖都会导入进来。要用什么功能就导入什么场景的启动器
4.3 主程序类,主入口类
@SpringBootApplication
Spring Boot应用标注在某个类上说明这个类是Spring Boot的主配置类,Spring Boot就应该运行这个类的main方法来启动Spring Boot应用
点击这个注解(annotation)查看源码
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(
excludeFilters = {@Filter(
type = FilterType.CUSTOM,
classes = {TypeExcludeFilter.class}
), @Filter(
type = FilterType.CUSTOM,
classes = {AutoConfigurationExcludeFilter.class}
)}
)
public @interface SpringBootApplication {
@AliasFor(
annotation = EnableAutoConfiguration.class,
attribute = "exclude"
)
Class<?>[] exclude() default {};
@AliasFor(
annotation = EnableAutoConfiguration.class,
attribute = "excludeName"
)
String[] excludeName() default {};
@AliasFor(
annotation = ComponentScan.class,
attribute = "basePackages"
)
String[] scanBasePackages() default {};
@AliasFor(
annotation = ComponentScan.class,
attribute = "basePackageClasses"
)
Class<?>[] scanBasePackageClasses() default {};
}
解析
@SpringBootConfiguration // spring boot定义的注解
Spring Boot的配置类,标注在某个类上,表示这是一个Spring Boot的配置类
点击这个注解(annotation)查看源码
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Configuration
public @interface SpringBootConfiguration {
}
解析
@Configuration // spring定义的注解
配置类上来标注这个注解
配置类---配置文件:配置类也是容器中的一个组件:
@Commponent
@EnableAutoConfiguration
开启自动配置功能
以前我们需要配置的东西,Spring Boot帮我们自动配置,这个注解告诉Spring Boot开启自动配置功能,这样自动配置才能生效
点击这个注解(annotation)查看源码
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import({EnableAutoConfigurationImportSelector.class})
public @interface EnableAutoConfiguration {
String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";
Class<?>[] exclude() default {};
String[] excludeName() default {};
}
解析
@AutoConfigurationPackage
自动配置包
点击这个注解(annotation)查看源码
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@Import({Registrar.class})
public @interface AutoConfigurationPackage {
}
解析
@Import({Registrar.class})
Spring的底层注解@import
,给容器中导入一个组件
导入的组件由Registrar.class
static class Registrar implements ImportBeanDefinitionRegistrar, DeterminableImports {
...
// metadata是注解的原信息
public void registerBeanDefinitions(AnnotationMetadata metadata, BeanDefinitionRegistry
registry) {
AutoConfigurationPackages.register(registry, (new AutoConfigurationPackages.PackageImport(
metadata)).getPackageName());
}
...
}
将主配置类(
@SpringBootApplication
标注的类)的所在包及下面所有子包里面的所有组件扫描到Spring容器
@Import({EnableAutoConfigurationImportSelector.class})
给容器中导入组件,这是导入那些组件的选择器
点击这个注解(annotation)查看源码
/** @deprecated */
@Deprecated
public class EnableAutoConfigurationImportSelector extends AutoConfigurationImportSelector {
public EnableAutoConfigurationImportSelector() {
}
protected boolean isEnabled(AnnotationMetadata metadata) {
return this.getClass().equals(EnableAutoConfigurationImportSelector.class) ? (Boolean
)this.getEnvironment().getProperty("spring.boot.enableautoconfiguration", Boolean.class, true) : true;
}
}
然后再查看其父类的源码
里面有一个selectImports
方法,将所有需要导入的组件以全类名的方式返回,这些组件就会被添加到容器中
会给容器中导入非常多的自动配置类(xxxAutoConfiguration):就是给容器中这个场景所需要的所有组件,并配置好这些组件
有了自动配置类,免去了我们手动编写配置注入功能组件等的工作
selectImports
会调用该类里的getCandidateConfigurations
方法
getCandidateConfigurations
又调用了SpringFactoriesLoader.loadFactoryNames(EnableAutoConfiguration.class, classLoader)
Spring Boot在启动的时候从类路径下的META-INF/spring.factories
中获取EnableAutoConfiguration
指定的值,将这些值作为自动配置类导入到容器中,自动配置类就生效,帮我们进行自动配置工作
以前我们需要自己配置的东西,自动配置类帮我们
J2EE的整体整合解决方案和自动配置都在spring-boot-autoconfigure-1.5.9.RELEASE.jar
5、使用Spring Initializer快速创建Spring Boot项目
IDE都支持使用Spring的项目创建向导快速创建一个Spring Boot项目;
选择我们需要的模块,向导会联网创建Spring Boot项目;
默认生成的Spring Boot项目:
-
主程序已经生成好了,我们只需要添加自己的逻辑即可
-
resources文件夹中的目录结构
- static:保存所有的静态资源:js, css, images
- templates:保存所以的模板页面:(Spring Boot默认jar包使用嵌入式的Tomcat,默认不支持JSP页面),可以使用官方推荐模板引擎(Freemarker, Thymeleaf)
- application.properties:Spring Boot应用的配置文件,虽然Spring Boot有默认配置,但我们可以修改一些默认配置