Springboot 系列(十五)如何编写自己的 Springboot starter
1. 前言
Springboot
中的自动配置确实方便,减少了我们开发上的复杂性,那么自动配置原理是什么呢?之前我也写过了一篇文章进行了分析。
Springboot 系列(三)Spring Boot 自动配置。
由于自动配置用到了配置文件的绑定,如果你还不知道常见的配置文件的用法,可以参考这篇文章。
Springboot 系列(二)Spring Boot 配置文件。
在这一次,通过学习 Springboot
自动配置模式,编写一个自己的 starter
,用来加深对自动配置的理解。
熟悉模式,有助于提升编写的 starter
的规范性,编写自己的 starter
之前先来学习 Springboot
官方 starter
以及常见框架的整合 starter
的编写方式 ,可以领略到其中的奥秘。
2. Springboot 官方模式
选择一个官方的自动配置进行分析,这里就选择常见的配置端口号配置。
2.1. 引入依赖
使用端口号之前我们需要先引入 web 依赖。
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
如果你观察 starter
多的话,也许你发已经发现了一个模式,Springboot
官方的 starter
的名字都是 spring-boot-starter-xxxx
命名的。
查看 spring-boot-starter-web
会发现,其实这个依赖只是一个空盒子,除了依赖其他 pom
之外,没有一行代码。
这时,发现了另外一个模式:starter
只依赖其他 pom
,不做代码实现。
那么 spring-boot-starter-web
到底依赖了哪些内容?
观察这个依赖信息,然后再参照其他的官方 starter
,可以找到几个固定的引入,可以被称之为模式的依赖引入。
- 依赖
spring-boot-starter
。 - 依赖
spring-boot-autoconfigure
。
2.2. 自动配置
引入依赖只有配置端口号,像这样。
server.port=8090
IDEA 中可以通过点击 server.port
找到这个配置绑定的类文件。可以看到配置最终会注入到类ServerProperties
类的 port
属性上。
那么这个 ServerProperties
到底是哪里使用的呢?继续查找,找到一个和 Servlet
的有关的调用。
发现是被 ServletWebServerFactoryCustomizer
类进行了调用,这个类里面定义了
private final ServerProperties serverProperties;
用来使用配置的属性。
继续查看这个类的调用,发现只有一个类使用这个类,这个类是ServletWebServerFactoryAutoConfiguration
。
根据我们对注解的理解,这个类就是自动配置主要类了。同时自动配置类都是以 AutoConfiguration
结尾。
看这个类的几个注解的意思。
- 优先级别较高。
@AutoConfigureOrder(Ordered.HIGHEST_PRECEDENCE)
- 只有在
ServletRequest
类存在和是 Web 应用时生效。
@ConditionalOnClass(ServletRequest.class)
@ConditionalOnWebApplication(type = Type.SERVLET)
- 开启了
ServerProperties
的配置绑定。
@EnableConfigurationProperties(ServerProperties.class)
- 导入了几个类。
@Import({ ServletWebServerFactoryAutoConfiguration.BeanPostProcessorsRegistrar.class,
ServletWebServerFactoryConfiguration.EmbeddedTomcat.class,
ServletWebServerFactoryConfiguration.EmbeddedJetty.class,
ServletWebServerFactoryConfiguration.EmbeddedUndertow.class })
同时注入配置到 Bean 工厂以供其他地方调用。
@Bean
public ServletWebServerFactoryCustomizer servletWebServerFactoryCustomizer(ServerProperties serverProperties) {
return new ServletWebServerFactoryCustomizer(serverProperties);
}
自动配置仅仅是这些东西吗?根据之前文章里的分析,我们知道不止代码,至少还有一个指定自动配置类的配置文件需要读取。也就是 spring.factories
文件。
如果你不知道,可以先看这篇文章。Springboot 系列(三)Spring Boot 自动配置 。
事实确实如此,可以在 spring.factories
中找到上面跟踪到的类。
也就是 ServletWebServerFactoryAutoConfiguration
.
根据上面的分析,可以发现 Springboot
官方 starter
的几个模式。
- 使用
XXXProperties
自动绑定XXX
开头的配置信息,如:ServerProperties
。 - 把
XXXProperties
定义到要使用的类中,如:ServletWebServerFactoryCustomizer
。 - 编写一个
XXXAutoConfiguration
,开启XXXProperties
的自动配置,限定生效场景,创建需要的类到Bean
工厂。如:ServletWebServerFactoryAutoConfiguration
。
3. 第三方集成模式
Springboot
官方如果把所有的框架都编写成 starter
,是不现实的。因此很多第三方框架需要主动集成到 springboot
,所以我们选择一个常用的框架分析它的 starter
实现。因为已经看过了 springboot
官方 starter
是如何配置的, 第三方框架也是类似,所以在下面观察的过程中会直接指出相同点,而不再做对比详细对比。
这里选择 mybatis-spring-boot-starter
进行学习分析。
3.1 引入依赖
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>1.3.2</version>
</dependency>
这里 mybatis
框架的 starter
依赖符合一定的规则,即 xxx-spring-boot-starter.
观察这个 starter
,发现它也没有做任何的代码实现,这一点和 springboot
官方一致。
查看 mybatis-spring-boot-starter
的依赖项,有很多,其中和自动配置有关的主要是。
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-autoconfigure</artifactId>
</dependency>
3.2 自动配置
查看 mybatis-spring-boot-autoconfigure
的内容发现和 springboot
官方的 autoconfigure
结构上是差不多的。
mybatis
的自动配置也是通过 spring.factories
来指明自动配置,然后通过 XxxAutoConfiguration
绑定 XxxProperties
来进行自动配置.
在原理上,和上面 springboot
官方的 starter
是相同的,所以不做过多的介绍了。
4. 编写自己的 starter
说了那么多,终于到了实操环节,通过上面的介绍,我们可以大致得出编写自己的 starter
步骤。
- 创建名字为
xxx-spring-boot-starter
的启动器项目。 - 创建名字为
xxx-spring-boot-autoconfigure
的项目。- 编写属性绑定类
xxxProperties
. - 编写服务类,引入
xxxProperties
. - 编写自动配置类
XXXAutoConfiguration
注入配置。 - 创建
spring.factories
文件,用于指定要自动配置的类。
- 编写属性绑定类
- 启动器项目为空项目,用来引入
xxx-spring-boot-autoconfigure
等其他依赖。 - 项目引入
starter
,配置需要配置的信息。
4.1 创建启动器项目
由于启动器不需要代码实现,只需要依赖其他项目,所以直接创建一个空的 maven 项目。但是名字要规范。
这里创建的 starter
是 myapp-spring-boot-starter
。
pom 文件非常简单,只需要引入接下来要创建的 myapp-spring-boot-autoconfigure
.
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>net.codingme.starter</groupId>
<artifactId>myapp-spring-boot-starter</artifactId>
<version>1.0-SNAPSHOT</version>
<!-- 启动器 -->
<dependencies>
<!-- 引入自动配置项目 -->
<dependency>
<groupId>net.codingme.starter</groupId>
<artifactId>myapp-spring-boot-autoconfigure</artifactId>
<version>0.0.1-SNAPSHOT</version>
</dependency>
</dependencies>
</project>
4.2 创建自动配置项目
结合上面对 starter
的分析,直接创建一个名字为 myapp-spring-boot-autoconfigure
的项目。项目中只引入 springboot
父项目以及 spring-boot-starter
。
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.2.0.RELEASE</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
<groupId>net.codingme.starter</groupId>
<artifactId>myapp-spring-boot-autoconfigure</artifactId>
<version>0.0.1-SNAPSHOT</version>
<name>myapp-spring-boot-autoconfigure</name>
<description>Demo project for Spring Boot</description>
<properties>
<java.version>1.8</java.version>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
</dependencies>
</project>
项目的总体结构看图。
在 HelloProperties
中通过注解 @ConfigurationProperties(prefix = "myapp.hello")
让类中的属性与 myapp.hello
开头的配置进行绑定。
/**
* <p>
*
* @Author niujinpeng
* @Date 2019/10/29 23:51
*/
@ConfigurationProperties(prefix = "myapp.hello")
public class HelloProperties {
private String suffix;
public String getSuffix() {
return suffix;
}
public void setSuffix(String suffix) {
this.suffix = suffix;
}
}
然后在 HelloService
中的 sayHello
方法使用 HelloProperties
中自动绑定的值。
public class HelloService {
HelloProperties helloProperties;
public String sayHello(String name) {
return "Hello " + name + "," + helloProperties.getSuffix();
}
public HelloProperties getHelloProperties() {
return helloProperties;
}
public void setHelloProperties(HelloProperties helloProperties) {
this.helloProperties = helloProperties;
}
}
为了让 HelloService
可以自动注入且能正常使用 HelloProperties
,所以我们在
HelloServiceAutoConfiguration
类中把 HelloProperties.class
引入,然后把 HelloService
注入到 Bean
。
/**
* web应用才生效
*/
@ConditionalOnWebApplication
/**
* 让属性文件生效
*/
@EnableConfigurationProperties(HelloProperties.class)
/***
* 声明是一个配置类
*/
@Configuration
public class HelloServiceAutoConfiguration {
@Autowired
private HelloProperties helloProperties;
@Bean
public HelloService helloService() {
HelloService helloService = new HelloService();
helloService.setHelloProperties(helloProperties);
return helloService;
}
}
最后在 spring.factories
中只需要指定要自动配置的类即可。
# Auto Configure
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
net.codingme.starter.HelloServiceAutoConfiguration
到这里,自动配置项目就完成了。可以在 myapp-spring-boot-autoconfigure
项目执行 mvn install
把自动配置项目打包到本地仓库,然后使用相同的命令把 myapp-spring-boot-starter
安装到仓库。因为后者依赖于前者项目,所以这里前者需要先进 mvn install
。
4.3 使用自定义的启动器
创建一个 springboot
项目myapp-spring-boot-starter-test
。
引入 web
依赖,引入自己编写的 myapp-spring-boot-starter
.
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!-- 引入自己的 starter -->
<dependency>
<groupId>net.codingme.starter</groupId>
<artifactId>myapp-spring-boot-starter</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
编写一个 HelloController
注入自动配置里的 HelloService
用于测试。
@RestController
public class HelloController {
@Autowired
HelloService helloService;
@GetMapping("/hello")
public String sayHello(String name) {
return helloService.sayHello(name);
}
}
由于 autoConfigure
项目中定义了 sayHello
方法会输出“Hello”+传入的 name + 配置的 hello.suffix
,所以我们在 springboot
配置文件中配置这个属性。
myapp.hello.suffix=早上好
运行测试项目,访问 /hello 路径传入一个 name 看看自动配置有没有生效。
从测试结果可以看到自动配置的早上好已经生效了。到这里自己编写的 starter
也已经完工。
项目已经传到 Github.
https://github.com/niumoo/springboot/tree/master/springboot-starter
<完>
我的微信:wn8398
个人主页:www.codingme.net
本篇文章是博主原创文章,欢迎转载,转载时在明显位置注明原文链接即可。
关注公众号回复资源可以获取Java 核心知识整理&面试资料。
Springboot 系列(十五)如何编写自己的 Springboot starter的更多相关文章
- springboot系列十五、springboot集成PageHelper
一.介绍 项目中经常会遇到分页,PageHelper为我们解决了这个问题.本质上实现了Mybatis的拦截器,作了分页处理. 二.配置PageHelper 1.引入依赖 pagehelper-spri ...
- SpringBoot系列(十二)过滤器配置详解
SpringBoot(十二)过滤器详解 往期精彩推荐 SpringBoot系列(一)idea新建Springboot项目 SpringBoot系列(二)入门知识 springBoot系列(三)配置文件 ...
- SpringBoot系列(五)Mybatis整合完整详细版
SpringBoot系列(五)Mybatis整合 目录 mybatis简介 项目创建 entity dao service serviceImpl mapper controller 1. Mybat ...
- 学习ASP.NET Core Razor 编程系列十五——文件上传功能(三)
学习ASP.NET Core Razor 编程系列目录 学习ASP.NET Core Razor 编程系列一 学习ASP.NET Core Razor 编程系列二——添加一个实体 学习ASP.NET ...
- SpringBoot | 第二十五章:日志管理之自定义Appender
前言 前面两章节我们介绍了一些日志框架的常见配置及使用实践.一般上,在开发过程中,像log4j2.logback日志框架都提供了很多Appender,基本上可以满足大部分的业务需求了.但在一些特殊需求 ...
- SpringBoot | 第十五章:基于Postman的RESTful接口测试
前言 从上一章节开始,接下来的几个章节会讲解一些开发过程中配套工具的使用.俗话说的好,工欲善其事,必先利其器.对于开发人员而言,有个好用的工具,也是一件事半功倍的事,而且开发起来也很爽,效率也会提升很 ...
- SpringBoot第十五篇:swagger构建优雅文档
作者:追梦1819 原文:https://www.cnblogs.com/yanfei1819/p/11007470.html 版权声明:本文为博主原创文章,转载请附上博文链接! 引言 前面的十四 ...
- 聊聊MySQL的加锁规则《死磕MySQL系列 十五》
大家好,我是咔咔 不期速成,日拱一卒 本期来聊聊MySQL的加锁规则,知道这些规则后可以判断SQL语句的加锁范围,同时也可以写出更好的SQL语句,防止幻读问题的产生,在能力范围内最大程度的提升MySQ ...
- 【Qt编程】基于Qt的词典开发系列<十五>html特殊字符及正则表达式
1.html特殊字符的显示 我们知道html语言和C语言一样也有一些特殊字符,它们是不能正常显示的,必须经过转义,在网上可以查到如何显示这些字符,如下图所示: 上图给了最常用的特殊字符的显示,下面我们 ...
- Springboot 系列(五)Spring Boot web 开发之静态资源和模版引擎
前言 Spring Boot 天生的适合 web 应用开发,它可以快速的嵌入 Tomcat, Jetty 或 Netty 用于包含一个 HTTP 服务器.且开发十分简单,只需要引入 web 开发所需的 ...
随机推荐
- Python 爬虫监控女神的QQ空间新的说说,实现邮箱发送
主要实现的功能就是:监控女神的 QQ空间,一旦女神发布新的说说,你的邮箱马上就会收到说说内容,是不是想了解一下 先看看代码运行效果图: PS:只有你有一台云服务器你就可以把程序24h运行起来 直接上代 ...
- SpringBootSecurity学习(03)网页版登录添加自定义登录页面
自定义登录页面 前面无论是使用默认配置,还是自定义配置类,都是使用的springboot-security自带的登录页面,自带的登录页面在这个版本虽然设计的非常不错,但是在实际开发中,我们通常还是使用 ...
- 从壹开始学习NetCore 45 ║ 终于解决了事务问题
一.项目说明 哈喽,又来写文章了,原来放假可以这么爽,可以学习和分享,
- postman工具使用小结
序言 现在,postman在做接口测试方面,发挥着越来越重大的作用,其支持多种请求方式.并可以模拟各种类型的数据请求类型,在实际开发中使用它可以极大的提高开发的效率. 安装postman 1. 安装 ...
- 06-border
边框 border:边框,描述盒子的边框 边框的三要素:粗细 线性样式 颜色 例如:border:1px solid red: 如果颜色不写,默认是黑色:如果粗细不写,不显示边框:如果只写线性样式,默 ...
- Django之使用redis缓存session,历史浏览记录,首页数据实现性能优化
Redis缓存session 配置Django缓存数据到redis中 # diango的缓存配置 CACHES = { "default": { "BACKEND&quo ...
- VR应用评测 - Google Spotlight Story: Sonaria
Google Spotlight Story: Sonaria 一个5min左右的VR小电影,坐姿观看,但是用户其实可以移动+旋转视角.画面很抽象,所有的物体都由基本的单色几何形状组成,主角是两个一公 ...
- 队列 & 栈---概述
队列 是一种特殊的线性表,特殊之处在于它只允许在表的前端(front)进行删除操作,而在表的后端(rear)进行插入操作,和栈一样,队列是一种操作受限制的线性表.进行插入操作的端称为队尾,进行删除操作 ...
- 即学即用的 30 段 Python 实用代码
[☞ 分享:最全最新的Python学习大礼包 ☜ 点击查看](https://mp.weixin.qq.com/s?__biz=MzU2MzgyODA4OA==&mid=100000592&a ...
- B-线性代数-范数
目录 范数 一.Lp范数 二.L0范数 三.L1范数 四.L2范数 五.L∞范数 更新.更全的<机器学习>的更新网站,更有python.go.数据结构与算法.爬虫.人工智能教学等着你:ht ...