spring项目中starter包的原理,以及自定义starter包的使用
MAVEN项目中starter的原理
一.原始方式
我们最早配置spring应用的时候,必须要经历的步骤:1.pom文件中引入相关的jar包,包括spring,redis,jdbc等等 2.通过properties或者xml配置相关的信息 3.不断调试直到可以使用。
问题:时间长,复杂,同时在写下一个项目的时候大概率要经过相同的模式配置才能达到可以使用的状态。同时在众多的jar中,我们需要相互配置依赖间的版本关系,十分的复杂
原始版本:

我们就想到能不能把这些jdbc整合起来,类似于深度学习中anaconda下载依赖一样去管理,依赖间的关系不需要我们去负责,而是交给spring去管理。
starter版本:

我们可以将starter包看作是一个包装箱,把复杂的事情都交给了spring负责,官方维护starter包会导入的东西。而我们只需要知道那个starter包是有什么用处,例如:spring-boot-starter-web是负责spring web项目的依赖。
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
二.starter内部详情

starter文件也只是一个pom文件,而不是jar,它的目的也是去自动的引入其他的jar文件,上图展示的spring-boot-starter-web中的依赖就有spring-boot-starter。

starter只是一个pom文件
下面也就是starter的关键所在,请问我们为什么引入了starter之后只需要配置一点点的个性化设置,例如创建application.properties仅仅配置端口等等就可以完成启动应用?是谁帮助我们配置了其他复杂的信息?
引出自动配置
三.自动配置
1.自动配置类的梳理
自动配置主要通过xxxAutoConfiguration这些类来实现,我们查找一个这样的类来进行示例演示

上图是DataSourceAutoConfiguration这个自动配置类,我们可以看到类上的几个注解。
- @Configuration 将该类标记为配置类,@Configuration注解的类可以看作是能生产让Spring IoC容器管理的Bean实例的工厂
- @ConditionalOnClass表示某个类位于类路径上时候,才会实例化这个bean
- @EnableConfigurationProperties注解的作用是使@ConfigurationProperties注解生效。如果只配置@ConfigurationProperties注解,在spring容器中是获取不到yml或者properties配置文件转化的bean的。
我们点击@EnableConfiguration注解中的@DataSourceProperties进去查看

可以查看到这里使用@configurationPropertes,@ConfigurationProperties注解的作用是把yml或者properties配置文件转化为bean。
同时这里也设置了prefix前缀,在我们项目的application.properties中配置的时候,提示的就是这些bean实例中的属性。

所以是使用@ConfigurationProperties和@EnableConfigurationProperties这两个注解来完成将一个包含众多属性的类来注册成为可供springIoc容器管理的bean。
而这个bean的注册过程是在各个XXXAutoConfiguration类中完成的。
2.如何发现依赖包中的bean
我们都知道springboot默认扫描启动类下面的主类和子类的bean来完成注解,但是并没有包括依赖包中的类,那么依赖包中的bean是如何被发现和加载的?
关键在于@SpringBootApplication这个注解
注解层次:
@springbootApplication
@SpringBootConfiguration:和@Configuration相同的用处,并且将里面通过@bean注解标注的方法的返回值作为bean对象注册到ioc容器之中。
获得bean的两种方式,一种是在配置类中通过方法返回,一种是直接在类上注解@bean(或者相同的注解,类似@Mapper等)来注册为bean实例
@EnableAutoConfiguration:借助@Import的支持,收集和注册依赖包中相关的bean定义。
@Import({AutoConfigurationImportSelector.class}):该注解扫描依赖包下需要注册为bean的自动配置类。
protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {
List<String> configurations = SpringFactoriesLoader.loadFactoryNames(this.getSpringFactoriesLoaderFactoryClass(), 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;
}
SpringFactoriesLoader.loadFactoryNames方法调用loadSpringFactories方法从所有的jar包中读取META-INF/spring.factories文件信息。
而Spring.factories中key/value中就有一个key是:org.springframework.boot.autoconfigure.EnableAutoConfiguration,后面跟着的都是需要AutoConfigurationImportSelector来进行注册的自动配置类

- @AutoConfigurationPackage
- @Import({Registrar.class}):Registrar就是扫描启动类目录下的所有bean并且注册,具体实现通过下列代码:
~~~ java
static class Registrar implements ImportBeanDefinitionRegistrar, DeterminableImports {
Registrar() {
}
public void registerBeanDefinitions(AnnotationMetadata metadata, BeanDefinitionRegistry registry) {
AutoConfigurationPackages.register(registry, (new AutoConfigurationPackages.PackageImport(metadata)).getPackageName());
}
public Set<Object> determineImports(AnnotationMetadata metadata) {
return Collections.singleton(new AutoConfigurationPackages.PackageImport(metadata));
}
}
~~~
注解层次图示:

3.如何加载发现的bean
如果要让一个普通类交给Spring容器管理,通常有以下方法:
- 使用 @Configuration与@Bean 注解
- 使用@Controller @Service @Repository @Component 注解标注该类,然后启用@ComponentScan自动扫描
- 使用@Import 方法
springboot中使用了@Import 方法
@EnableAutoConfiguration注解中使用了@Import({AutoConfigurationImportSelector.class})注解,AutoConfigurationImportSelector实现了DeferredImportSelector接口,
DeferredImportSelector接口继承了ImportSelector接口,ImportSelector接口只有一个selectImports方法。
selectImports方法返回一组bean,@EnableAutoConfiguration注解借助@Import注解将这组bean注入到spring容器中,springboot正式通过这种机制来完成bean的注入的。
关于@import注解的加载可以查看这个文章:https://zhuanlan.zhihu.com/p/147025312
ps:晕乎乎的,我只看懂了一部分。
加载redisAutoConfiguration的流程图示

四.自定义starter
下面我们演示自定义starter的流程:

我们确定好自定义starter的GAV(groupId,ArtifactId,Version),这里需要注意的是ArtifactId的命名,对于spring进行管理的starter包,命名规则是:spring-boot-starter-xxx,而为了区别spring进行管理的starter包,自定义的starter包一般命名规则是:xxx-spring-boot-starter,例如mybatis官方推出的starter包:mybatis-spring-boot-starter。
groupId使用自己域名的反写即可。
项目结构如下:

1.pom文件导入依赖
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
<version>2.7.3</version>
<optional>true</optional>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
<version>2.7.1</version>
<optional>true</optional>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.24</version>
</dependency>
</dependencies>
依赖:spring-boot-configuration-processor 作用:导入之后会自动生成元数据(meta-data),在application.properties配置的时候会有提示。
例如:

由于是自动生成,在你配置了@ConfigurationProperties之后就会自动生成元数据,在你写application.properties的时候就会进行自动提示。

当无法提示的时候多次clean,之后再compile然后在install发布。
具体可以看这个博客 https://blog.csdn.net/wangleleb/article/details/104904348
依赖: spring-boot-starter 作用:是为了使用前面提到的自动配置的注解。注解@ConfigurationProperties和@EnableConfigurationProperties两个注解都在spring-boot-context包中
2.编写代码
编写自动配置类代码:
package properties;
import lombok.Getter;
import lombok.Setter;
import org.springframework.boot.context.properties.ConfigurationProperties;
@ConfigurationProperties(prefix = "demo")
@Getter
@Setter
public class DemoProperties{
private String var1;
private String var2;
}
编写service代码:
package service;
import lombok.AllArgsConstructor;
@AllArgsConstructor
public class DemoService {
public String var1;
public String var2;
public String variable(){
return this.var1 + " " + this.var2;
}
}
编写config类:
@Configuration
@EnableConfigurationProperties(DemoProperties.class)
// 只有当name的值与havingValue的值相同的时候加载
@ConditionalOnProperty(
prefix = "demo",
name = "isopen",
havingValue = "true"
)
public class DemoConfig {
@Resource
private DemoProperties demoProperties;
@Bean
public DemoService demoService(){
return new DemoService(demoProperties.getVar1(),demoProperties.getVar2());
}
}
关于这里@ConditionOnProperty这个注解,该注解的大概含义就是,对于前缀是demo的属性,底下的值isopen为true时候,该自动配置类才会生效。
编写spring.factories
将key:EnableAutoConfiguration-->DemoConfig这个类

3.打包
使用maven命令:mvn clean compile install 清理,编译,发布到本地仓库中去
4.其他项目引入
<!--引入我写的starter-->
<dependency>
<groupId>org.oldoldcoder</groupId>
<artifactId>oldoldcoder-spring-boot-starter</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
配置资源文件

# 使用自己写的starter
demo.isopen=true
demo.var1=var1
demo.var2==var2
随便编写一个类验证
@Component
public class TestService {
@Resource
private DemoService demoService;
@PostConstruct
public void test(){
System.out.println("你好"+demoService.variable());
}
}
结果

spring项目中starter包的原理,以及自定义starter包的使用的更多相关文章
- spring框架中@PostConstruct的实现原理
在spring项目经常遇到@PostConstruct注解,首先介绍一下它的用途: 被注解的方法,在对象加载完依赖注入后执行. 此注解是在Java EE5规范中加入的,在Servlet生命周期中有一定 ...
- java web项目(spring项目)中集成webservice ,实现对外开放接口
什么是WebService?webService小示例 点此了解 下面进入正题: Javaweb项目(spring项目)中集成webservice ,实现对外开放接口步骤: 准备: 采用与spring ...
- spring 项目中在类中注入静态字段
有时spring 项目中需要将配置文件的属性注入到类的静态字段中 例如:文件上传 //文件上传指定上传位置 //resource-dev.properties 有如下参数 #upload UPLOAD ...
- 在 ASP.NET Core 项目中使用 npm 管理你的前端组件包
一.前言 在项目的前端开发中,对于绝大多数的小伙伴来说,当然,也包括我,不可避免的需要在项目中使用到一些第三方的组件包.这时,团队中的小伙伴是选择直接去组件的官网上下载,还是图省事直接在网上搜索,然后 ...
- 【SpringBoot1.x】SpringBoot1.x 启动配置原理 和 自定义starter
SpringBoot1.x 启动配置原理 和 自定义starter 启动配置原理 本节源码 启动过程主要为: new SpringApplication(sources) 创建 SpringAppli ...
- spring项目中使用定时任务
当我们希望在某个时间点来执行一些业务方法的时候就用到定时任务,在spring的项目中使用定时任务很简单.如下 第一步.加入jar包 <dependency> <groupId> ...
- spring项目中如何添加定时器以及在定时器中自动生成sprng注入对象
最近做了一个java的项目,部门领导给了一套代码让我尽快掌握,说心里话本人真心不喜欢java的这种项目方式,各种配置各种xml文件简直头都大了,下面就将我遇到的其中一个我认为是坑的地方整理出来,希望能 ...
- Spring项目中Properties不能加载多个的问题
A模块和B模块都分别拥有自己的Spring XML配置,并分别拥有自己的配置文件: A模块 A模块的Spring配置文件如下: <?xml version="1.0" enc ...
- 【主流技术】ElasticSearch 在 Spring 项目中的实践
前言 ElasticSearch简称es,是一个开源的高扩展的分布式全文检索引擎. 它可以近乎实时的存储.检索数据,其扩展性很好,ElasticSearch是企业级应用中较为常见的技术. 下面和大家分 ...
随机推荐
- 搭建ceph分布式文件系统
1. 准备4台虚拟机 ceph 192.168.66.93 管理osd,mon节点 ceph-node1 192.168.66.94 osd节点 ceph-node2 192.168.66.95 ...
- cut命令、case与select语句
cut命令 常用参数: -c character 字符 -d delimiter 分隔符 -f field 域(列) --output-delimiter 输出分隔符 例: # echo 12345 ...
- 浪姐打分看不够?用几行Python代码模拟评委打分
大家好鸭~我是小熊猫比赛大家都看过吧,每次是不是都对比赛成绩充满期待.特别是浪姐的打分看的简直欲罢不能- 今天就用Python来模拟评委打分,这个案例很短也很简单,很适合新手跟小白练习. 在某次十佳歌 ...
- Java中将对象或者集合对象转换成json字符串
1.对象和字符串相互转换 2.集合对象和字符串相互转换
- nifi从入门到实战(保姆级教程)——flow
本文章首发于博客园,转载请标明出处 经过前两篇文章(环境篇,身份验证),我们已经有了nifi可以运行的基础,今天就来实现一个案例吧. 假设我们要从ftp上获取一个zip包,里面有两个csv文件,一个是 ...
- 什么是pytorch?
Pytorch是基于python的科学计算包,为两类受众提供服务 作为Numpy的替换,让你可以使用GPU的算力 作为一个深度学习计算平台提供最大的计算灵活性与速度 开始体验pytorch的基础功能 ...
- Java学习dayo4
分支结构和循环语句 1.包的概念 包就是文件夹 包的命名规范:全小写,域名倒置,不能以点开头或结尾,可以包含点,每存在一个点表示一个子目录 举例:com.baidu.demo 定义包后,包中的java ...
- 从零开始制作【立体键盘】,画UI免写CSS,【盲打练习】的交互逻辑只用了10来行表达式!
手把手教你从空白页面开始通过拖拉拽可视化的方式制作[立体键盘]的静态页面,不用手写一行CSS代码,全程只用10来行表达式就完成了[盲打练习]的交互逻辑. 整个过程在众触应用平台进行,快速直观. 最终U ...
- 01 开发App真机调试问题
逍遥安卓模拟器 :https://juejin.cn/post/7062922018710093831 HBuilderX真机调试插上手机却提示"未检测到手机或浏览器"的问题:ht ...
- 图像处理——相位恢复(GS,TIE,改进型角谱迭代法)
利用GS,TIE,改进型角谱迭代算法进行相位恢复 角谱传播理论 角谱传播理论可以翻阅傅里叶光学的书,就能找到定量分析的计算公式,可以分析某个平面的角谱垂直传播到另外一个平面的角谱,得到其振幅与相位信息 ...