《深入Spring Boot》第一篇快速入门,本篇主要包含以下5部分内容:
Spring Boot是什么;
为什么要使用Spring Boot;
工具准备;
快速上手;
代码解析。
Spring Boot是什么
Spring Boot是Spring大家庭的成员,它是一个全新的框架。它的设计目的是尽可能简单和快速的开发、运行Spring应用程序,简化配置。
为什么要使用Spring Boot
在使用Spring框架进行开发的过程中,需要配置很多Spring框架包的依赖,如spring-core、spring-bean、spring-context等,而这些配置通常都是重复添加的,而且需要做很多框架使用及环境参数的重复配置,如开启注解、配置日志等。Spring Boot致力于弱化这些不必要的操作,提供默认配置,当然这些默认配置是可以按需修改的,快速搭建、开发和运行Spring应用。
工具准备
在进行快速上手之前,准备开发工具。
1.本文内容是基于Spring Boot最新稳定版1.5.10,要安装JDK7或更高版本;
2.依赖管理的构建工具,可以安装Ant、Maven或Gradle,本文使用Maven;
3.代码编辑工具推荐使用IntelliJ IDEA。
快速上手
下面使用Spring Boot编写一个Web应用,编写完成后并运行。
1.在IDEA中新建项目,直接上图。
2.下图是新建好的项目目录结构。
3.打开pom.xml,添加web应用需要的依赖配置,配置如下。
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.5.10.RELEASE</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
</dependencies>
添加后的pom.xml如图。
4.使用mvn package命令或IDEA自带的可视化工具加载依赖,本文使用IDEA自带的可视化工具,双击package立即执行依赖加载。
5.在java目录下新建Example类,编写代码,具体代码如下。
import org.springframework.boot.*;
import org.springframework.boot.autoconfigure.*;
import org.springframework.web.bind.annotation.*;
@RestController
@EnableAutoConfiguration
public class Example {
@RequestMapping("/")
String home() {
return "Hello World!";
}
public static void main(String[] args) throws Exception {
SpringApplication.run(Example.class, args);
}
}
6.直接执行Example类中的main方法启动应用,应用启动时会使用内嵌的Tomcat服务器,同时使用默认8080端口对外提供服务。成功启动后会输出如下图的日志信息。
7.打开浏览器访问http://localhost:8080/,页面会显示Hello World!。
至此,一个简单的Web应用就编写完成并可以提供服务。
代码解析
首先是pom.xml中的配置,继承
spring-boot-starter-parent是快速使用Spring Boot的一种方式,也可以选择不继承。Spring Boot中的Starter是一个包含很多依赖描述的集合,只要添加一个Starter,这个Starter里面约定的依赖都会被添加到项目中,例如示例代码添加了spring-boot-starter-web,执行maven操作会下载web应用需要的依赖jar,如下图。
示例代码中使用了@RestController、@RequestMapping、@EnableAutoConfiguration和SpringApplication,其中@RestController和@RequestMapping是SpringMVC中的注解,@EnableAutoConfiguration和SpringApplication是Spring Boot中的。
@EnableAutoConfiguration注解从字面上理解是开启自动配置,有注解就有对应的处理类,先看一下这个注解的定义。
这个注解有两个属性,exclude用于标记哪些类不需要被自动配置,excludeName也用于标记哪些类不需要被自动配置,只不过它存的是类的名称。
从上图的红框中可以看到在@EnableAutoConfiguration注解的定义中使用了@Import注解,@Import注解是在Spring3.0添加的,使用@Import导入
EnableAutoConfigurationImportSelector.class,作用等同于在Spring的xml文件中使用导入,查看EnableAutoConfigurationImportSelector源码你会发现它就是用于处理@EnableAutoConfiguration注解的,不过在Spring Boot1.5版本开始被标记为弃用,可以使用AutoConfigurationImportSelector代替,也就是它的父类。
进入
AutoConfigurationImportSelector类中,重要的方法是selectImports。
执行AnnotationAttributes attributes = getAttributes(annotationMetadata)可以获取@EnableAutoConfiguration注解中exclude和excludeName属性值,然后把值封装在AnnotationAttributes中。
执行List configurations =
getCandidateConfigurations(annotationMetadata,attributes),将在classpath下jar包中META-INF目录下搜索spring.factories文件,然后在spring.factories文件中查找以org.springframework.boot.autoconfigure.EnableAutoConfiguration为key的值,这个值是一个列表,列表如下图。
接着就是使用exclude和excludeName属性值排除掉不需要自动配置的类,需要自动化配置的类会被执行,例如AopAutoConfiguration会自动配置Aop相关配置,使用JDK代理还是CGLIB代理。
到这里,自动配置还没完成,有一个问题还没弄明白,就是
AutoConfigurationImportSelector类的selectImports方法是在哪里被触发的?解决这个问题可以通过断点调试或IDEA里的查找调用来获取调用方,无论哪种方式都可以发现是ConfigurationClassParser类的parse方法间接调用的,在parse方法中完成了需要自动配置类的初始化,进而完成一些默认的配置。那这个parse方法又是在哪被触发调用的?同样可以通过上面的任意方法,查找到是ConfigurationClassPostProcessor类的postProcessBeanFactory方法中调用的,而这个ConfigurationClassPostProcessor实现了BeanDefinitionRegistryPostProcessor接口,而BeanDefinitionRegistryPostProcessor继承了BeanFactoryPostProcessor接口,熟悉Spring原理的都知道BeanFactoryPostProcessor接口中的postProcessBeanFactory方法会在Spring容器加载bean的定义文件之后,在bean实例化之前执行,而这一步是由SpringApplication的run方法触发的,在这个run方法中实现了Spring容器的启动及初始化。