Spring Boot Starters 究竟是怎么回事
Spring Boot 对比 Spring MVC 最大的优点就是使用简单,约定大于配置。不会像之前用 Spring MVC 的时候,时不时被 xml 配置文件搞的晕头转向,冷不防还因为 xml 配置上的一点疏忽,导致整个项目莫名其妙的不可用,顿感生活无所依恋,简称生无可恋。
这要归功于组成了 Spring Boot 的各种各样的 starters,有官方提供的,也有第三方开源出来。可以这么说,基本上你打算用的功能都可以找到,如果没有找到,那就再找一找。
用 Spring Boot 的功能组件(例如 spring-boot-starter-actuator、 spring-boot-starter-data-redis 等)的步骤非常简单,用著名的把大象放冰箱的方法来概括的话,有以下三步就可以完成组件功能的使用:
STEP 1
在 pom 文件中引入对应的包,例如:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
STEP 2
在应用配置文件中加入相应的配置,配置都是组件约定好的,需要查看官方文档或者相关说明。有些比较复杂的组件,对应的参数和规则也相应的较多,有点可能多大几十上百了。
STEP 3
以上两步都正常的情况下,我们就可以使用组件提供的相关接口来开发业务功能了。
那么 Spring Boot 为什么能做到如此简单易用呢,它内部是什么样的工作机制呢,不知道你有没有研究过。
以下是为了理解 Spring Boot 组件的实现机制而制作的一个 demo starter。理解其中的原理,对我们日后的工作有什么意义呢?
1. 遇到问题的时候,可以帮助我们更有头绪的排查问题;
2. 可以帮助我们正确的阅读源代码,组件的切入口在哪儿,配置属性是什么等等;
开始实现这个简单的 starter,这个 starter 并没有什么实际的功能,只是为了做个演示而已。
开始之前,我们要理解一下 spring boot starter 是什么呢?
实际上 starter 并不会包含多少功能代码,我们可以把它理解成一个「连接包」(我自己造的概念),按照这个概念来说:
它首先是一个包,一个集合,它把需要用的其他功能组件囊括进来,放到自己的 pom 文件中。
然后它是一个连接,把它引入的组件和我们的项目做一个连接,并且在中间帮我们省去复杂的配置,力图做到使用最简单。
实现一个 starter 有四个要素:
- starter 命名 ;
- 自动配置类,用来初始化相关的 bean ;
- 指明自动配置类的配置文件 spring.factories ;
- 自定义属性实体类,声明 starter 的应用配置属性 ;
好了,开始实现我们的 demo
1. 给 starter 起个名字
也就是我们使用它的时候在 pom 中引用的 artifactId。命名有有规则的,官方规定:
官方的 starter 的命名格式为 spring-boot-starter-{name} ,例如上面提到的 spring-boot-starter-actuator。
非官方的 starter 的命名格式为 {name}-spring-boot-starter,我们把自定的 starter 命名为 kite-spring-boot-starter,命名在 pom 文件里。
<groupId>kite.springcloud</groupId>
<artifactId>kite-spring-boot-starter</artifactId>
<packaging>jar</packaging>
<version>1.0-SNAPSHOT</version>
2. 引入自动配置包及其它相关依赖包
实现 starter 主要依赖自动配置注解,所以要在 pom 中引入自动配置相关的两个 jar 包
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-autoconfigure</artifactId>
</dependency>
除此之外,依赖的其他包当然也要引进来。
3. 创建 spring.factories 文件
在 resource/META-INF 目录下创建名称为 spring.factories 的文件,为什么在这里?当 Spring Boot 启动的时候,会在 classpath 下寻找所有名称为 spring.factories 的文件,然后运行里面的配置指定的自动加载类,将指定类(一个或多个)中的相关 bean 初始化。
例如本例中的配置信息是这样的:
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
kite.springcloud.boot.starter.example.KiteAutoConfigure
等号前面是固定的写法,后面就是我们自定义的自动配置类了,如果有多个的话,用英文逗号分隔开。
4. 编写自动配置类
自动配置类是用来初始化 starter 中的相关 bean 的。可以说是实现 starter 最核心的功能。
@Configuration
@ConditionalOnClass(KiteService.class)
@EnableConfigurationProperties(KiteProperties.class)
@Slf4j
public class KiteAutoConfigure { @Autowired
private KiteProperties kiteProperties; @Bean
@ConditionalOnMissingBean(KiteService.class)
@ConditionalOnProperty(prefix = "kite.example",value = "enabled", havingValue = "true")
KiteService kiteService(){
return new KiteService(kiteProperties);
}
}
代码非常简单,放眼望去,最多的就是各种注解。
@Configuration 这个不用解释,表示这是个自动配置类,我们平时做项目时也会用到,一般是用作读取配置文件的时候。
@ConditionalOnClass(KiteService.class) :只有在 classpath 中找到 KiteService 类的情况下,才会解析此自动配置类,否则不解析。
@EnableConfigurationProperties(KiteProperties.class):启用配置类。
@Bean:实例化一个 bean 。
@ConditionalOnMissingBean(KiteService.class):与 @Bean 配合使用,只有在当前上下文中不存在某个 bean 的情况下才会执行所注解的代码块,也就是当前上下文还没有 KiteService 的 bean 实例的情况下,才会执行 kiteService() 方法,从而实例化一个 bean 实例出来。
@ConditionalOnProperty:当应用配置文件中有相关的配置才会执行其所注解的代码块。
这个类的整体含义就是: 当 classpath 中存在 KiteService 类时解析此配置类,什么情况下才会在 classpath 中存在呢,就是项目引用了相关的 jar 包。并且在上下文中没有 KiteService 的 bean 实例的情况下,new 一个实例出来,并且将应用配置中的相关配置值传入。
5. 实现属性配置类
@Data
@ConfigurationProperties("kite.example")
public class KiteProperties { private String host; private int port;
}
配置类很简单,只有两个属性,一个 host ,一个 port 。配置参数以 kite.example 作为前缀。稍后我们在使用这个 starter 的时候会看到如何声明配置属性。
6. 实现相关功能类
也就是前面一直提到的 KiteService,其实严格来讲,这个业务功能类不应该放在 starter 中,应该放在单独的 jar 包里,但是此处 demo 非常简单,也就在这里写了。
@Slf4j
public class KiteService { private String host; private int port; public KiteService(KiteProperties kiteProperties){
this.host = kiteProperties.getHost();
this.port = kiteProperties.getPort();
} public void print(){
log.info(this.host + ":" +this.port);
}
}
7. 打包
通过 maven 命令将这个 starter 安装到本地 maven 仓库
mvn install
也可以通过 mvn package deploy
发布到你的私服或者发布到中央仓库。
上面已经完成了 starter 的开发,并安装到了本地仓库,然后就是在我们的项目中使用它了。
1. 创建项目,在 pom 中引用
<dependency>
<groupId>kite.springcloud</groupId>
<artifactId>kite-spring-boot-starter</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
2. 应用配置项
创建 application.yml ,配置如下:
server:
port:
kite:
example:
enabled: true # 开启才生效
host: 127.0.0.1
port:
3. 调用 KiteService 的服务方法
@RestController
@RequestMapping(value = "use")
public class UseController { @Autowired
private KiteService kiteService; @GetMapping(value = "print")
public void print(){
kiteService.print();
}
}
4. 启动服务,并访问接口
访问 /use/print 接口,会发现在日志中打印出了配置信息
2019-05-24 16:45:04.234 INFO 36687 --- [nio-3801-exec-1] k.s.boot.starter.example.KiteService : 127.0.0.1:3801
顺着上面的思路,我们来看一下官方的 starters 的结构。先来把 Spring Boot 从 github 上 clone 一份下来。用 idea 打开,可以看到项目结构如下
Spring-boot-starters 中就是官方提供的主要 starters,比如 jdbc、redis、security、web 等等。
我们拿 spring-boot-starter-data-redis 这个 starter 作为例子,来说一说官方是怎么组织项目结构的,以及阅读源码的顺序应该是怎样的。
1. 展开 Spring-boot-staters 下的 redis starter,我们看到目录结构如下
其中并没有 Java 代码,只有一个 spring.provides 文件,里面的内容如下:
provides: spring-data-redis,lettuce-core
意思就是说,本项目依赖 spring-data-redis 和 lettuce-core 这两个包,并且在 pom 文件中引用了。其目的就是告知使用者在引用此包的时候,不必再引用 provides 中的依赖包了。
2. 然后就是自动注解了,所有 stater 的自动注解类、属性配置类都放到了 spring-boot-autoconfigure 这个项目下
看到熟悉的 spring.factories 没有,前面我们自己实现过。这个内容比较多,我们只看 redis 相关的
org.springframework.boot.autoconfigure.data.redis.RedisAutoConfiguration,\
org.springframework.boot.autoconfigure.data.redis.RedisReactiveAutoConfiguration,\
org.springframework.boot.autoconfigure.data.redis.RedisRepositoriesAutoConfiguration
包含三个自动配置文件,然后顺着配置,我们找到所在 package
然后就可以开始阅读代码了。其他的 starter 也是同样的结构。
Spring Boot Starters 究竟是怎么回事的更多相关文章
- 你一直在用的 Spring Boot Starters 究竟是怎么回事
Spring Boot 对比 Spring MVC 最大的优点就是使用简单,约定大于配置.不会像之前用 Spring MVC 的时候,时不时被 xml 配置文件搞的晕头转向,冷不防还因为 xml 配置 ...
- Spring Boot Starters到底怎么回事?
前言 上周看了一篇.你一直在用的Spring Boot Starters究竟是怎么回事(https://www.cnblogs.com/fengzheng/p/10947585.html) 感觉终 ...
- Spring Boot Starters介绍
文章目录 Web Start Test Starter Data JPA Starter Mail Starter 结论 对于任何一个复杂项目来说,依赖关系都是一个非常需要注意和消息的方面,虽然重要, ...
- 54 个官方 Spring Boot Starters 出炉!别再重复造轮子了…….
在之前的文章,栈长介绍了 Spring Boot Starters,不清楚的可以点击链接进去看下. 前段时间 Spring Boot 2.4.0 也发布了,本文栈长再详细总结下最新的 Spring B ...
- Spring Boot Starters
Spring Boot Starters 摘自 https://www.nosuchfield.com/2017/10/15/Spring-Boot-Starters/ 2017-10-15 Spri ...
- Spring Boot Starters 列表
Spring Boot application starters 名称 描述 Pom spring-boot-starter 核心starter,包括自动配置支持,日志和YAML Pom spring ...
- Spring Boot Starters启动器
Starters是什么? Starters可以理解为启动器,它包含了一系列可以集成到应用里面的依赖包,你可以一站式集成Spring及其他技术,而不需要到处找示例代码和依赖包.如你想使用Spring J ...
- Spring Boot Starters是什么?
版权声明:该文转自: http://www.nosuchfield.com/2017/10/15/Spring-Boot-Starters/.版权归原创作者,在此对原作者的付出表示感谢! starte ...
- Sping Boot入门到实战之实战篇(二):一些常用功能的Spring Boot Starters
包含功能 阿里云消息服务MNS 阿里云消息队列服务(即时消息.延迟消息.事务消息) AOP日志 基于MyBatis通用Mapper及DRUID的数据库访问 dubbo支持 错误处理 七牛图片服务 re ...
随机推荐
- 逻辑回归2-scikit-learn参数介绍
1.1 scikit-learn参数介绍 1.1.1 导入 from sklearn.linear_model import LogisticRegression 1.1.2 版本 sci ...
- CDH的ntp时间同步
云服务器: ntpq -p ntpdate -u 10.52.255.1 #手动同步 自建NTP服务器: https://www.cnblogs.com/yinzhengjie/p/9480665. ...
- spring boot-7.日志系统
日志系统分为两部分,一部分是日志抽象层,一部分是日志实现层.常见的日志抽象层JCL,SLF4J,JBoss-Logging,日志实现层有logback,log4j,log4j2,JUL.日志抽象层的功 ...
- DAX/PowerBI系列 - 累计总计(Cumulative Total, Running Total)
DAX/PowerBI系列 - 累计总计(Cumulative Total) 2017/07/23 更新:B列公式(见最后) 2019/08/08 更新:在可视化数据的时候,一定要选择日期维度的日期列 ...
- Mysql数据库中的输入命令各类知识总结
1.链接数据库的命令---mysql-u root-p 回车,输入密码//在cmd上输入自己的账号密码 2.查看:show databases: 3.创建数据库:create +database+数据 ...
- 关于React中props与state的一知半解
props props英文翻译是道具的意思,我个人理解为参数,如果我们将react组件看作是一个函数,那么props便是函数接收外部数据所使用的参数.props具有以下特性: 1.不可变(只读性) p ...
- 1.javascript知识点总结
1.写javaScript的三种方式: 2.写javaScript的注意事项: ①严格区分字母的大小写: ②空格和换行.多余的空格会被忽略,可以将一行代码分成多行写: ③分号作为一个语句的结束: ④单 ...
- css3 伪类以及伪元素的特效
菱形
- iOS蓝牙4.0开发
文/starfox寒流(简书作者)原文链接:http://www.jianshu.com/p/974d165f78b5著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”. iOS 蓝牙4.0 ...
- 2019-11-29-dotnet-文件读写务必注意事项
title author date CreateTime categories dotnet 文件读写务必注意事项 lindexi 2019-11-29 08:34:43 +0800 2019-10- ...