徒手撸了一个Starter,同事直夸我666~

开发 前端
Spring Boot 将常见的开发功能,分成了一个个的starter,这样我们开发功能的时候只需要引入对应的starter,而不需要去引入一堆依赖了!starter可以理解为一个依赖组,其主要功能就是完成引入依赖和初始化配置。

[[396095]]

Spring Boot starter原理

Spring Boot 将常见的开发功能,分成了一个个的starter,这样我们开发功能的时候只需要引入对应的starter,而不需要去引入一堆依赖了!starter可以理解为一个依赖组,其主要功能就是完成引入依赖和初始化配置。Spring 官方提供的starter 命名规范为 spring-boot-starter-xxx ,第三方提供的starter命名规范为 xxx-spring-boot-starter 。

这里我们以 RocketMQ 的依赖 rocketmq-spring-boot-starter 来学习 starter的原理。

在项目中引入 rocketmq-spring-boot-starter 之后,实际上就引入了 rocketmq 的一些相关依赖。

在 rocketmq-spring-boot 中有一个自动装配的类RocketMQAutoConfiguration ,我截取了其中的一小段代码,一起来看看。

  1. @Configuration 
  2. @EnableConfigurationProperties(RocketMQProperties.class) 
  3. @ConditionalOnClass({MQAdmin.class}) 
  4. @ConditionalOnProperty(prefix = "rocketmq", value = "name-server", matchIfMissing = true
  5. @Import({MessageConverterConfiguration.class, ListenerContainerConfiguration.class, ExtProducerResetConfiguration.class, RocketMQTransactionConfiguration.class}) 
  6. @AutoConfigureAfter({MessageConverterConfiguration.class}) 
  7. @AutoConfigureBefore({RocketMQTransactionConfiguration.class}) 
  8.  
  9. public class RocketMQAutoConfiguration { 
  10.     private static final Logger log = LoggerFactory.getLogger(RocketMQAutoConfiguration.class); 
  11.  
  12.     public static final String ROCKETMQ_TEMPLATE_DEFAULT_GLOBAL_NAME = 
  13.         "rocketMQTemplate"
  14.  
  15.     @Autowired 
  16.     private Environment environment; 
  17.  
  18.     @Bean(destroyMethod = "destroy"
  19.     @ConditionalOnBean(DefaultMQProducer.class) 
  20.     @ConditionalOnMissingBean(name = ROCKETMQ_TEMPLATE_DEFAULT_GLOBAL_NAME) 
  21.     public RocketMQTemplate rocketMQTemplate(DefaultMQProducer mqProducer, 
  22.         RocketMQMessageConverter rocketMQMessageConverter) { 
  23.         RocketMQTemplate rocketMQTemplate = new RocketMQTemplate(); 
  24.         rocketMQTemplate.setProducer(mqProducer); 
  25.         rocketMQTemplate.setMessageConverter(rocketMQMessageConverter.getMessageConverter()); 
  26.         return rocketMQTemplate; 
  27.     } 
  • @Configuration 说明这是一个配置类,类中被@Bean注解了的方法,就是spring的一个bean,例如rocketMQTemplate。
  • @EnableConfigurationProperties,启用被@ConfigurationProperties的bean,这里引入了 RocketMQProperties 。

RocketMQProperties 就是需要在yml文件中写入的属性。

  1. @ConfigurationProperties(prefix = "rocketmq"
  2. public class RocketMQProperties { 
  3.  
  4.     private String nameServer; 
  5.  
  6.     private String accessChannel; 
  7.  
  8.     private Producer producer; 
  9.  
  10.     private Consumer consumer = new Consumer(); 

在Spring Boot项目启动的时候默认只会扫描下级目录下带 @Configuration 注解的类,那么像本文中提到的 RocketMQAutoConfiguration 是如何扫描的呢?其实项目启动的时候会去加载项目中所有的 spring.factories 文件,然后加载对应的配置类,因此我们就需要在 spring.factories 中只指定需要扫描的类。

原理搞明白了,接下来我们就简单实现一个自己的starter!这个starter的主要作用就是给一个对象尾部拼接一个字符串!

一、新建项目

新建一个名为 javatip-spring-boot-starter 的项目,并且引入下面的依赖

  1. <dependency> 
  2.     <groupId>org.springframework.boot</groupId> 
  3.     <artifactId>spring-boot-starter</artifactId> 
  4. </dependency> 

二、新增配置类

配置类对应的properties文件中的属性为javatip.name;

  1. @ConfigurationProperties(prefix = "javatip"
  2. public class JavatipPorperties { 
  3.  
  4.     private String name
  5.  
  6.     public String getName() { 
  7.         return name
  8.     } 
  9.  
  10.     public void setName(String name) { 
  11.         this.name = name
  12.     } 

三、新增拼接字符串的方法

此方法主要就是为对象拼接一个固定的字符串

  1. public class StrUt { 
  2.  
  3.     private String name
  4.  
  5.     public String strTo(Object object){ 
  6.  
  7.         return object +"---"+ getName(); 
  8.     } 
  9.  
  10.     public String getName() { 
  11.         return name
  12.     } 
  13.  
  14.     public void setName(String name) { 
  15.         this.name = name
  16.     } 

四、新增自动配置类

使用注解 @EnableConfigurationProperties 启用 JavatipProperties 配置类

使用注解 @Configuration 配合 @Bean 注册一个拼接字符串的bean对象。

  1. @Configuration 
  2. @EnableConfigurationProperties(JavatipPorperties.class) 
  3. public class JavatipAutoConfiguration { 
  4.  
  5.     @Autowired 
  6.     private JavatipPorperties javatipPorperties; 
  7.  
  8.     @Bean 
  9.     public StrUt strut(){ 
  10.         StrUt strut = new StrUt(); 
  11.         strut.setName(javatipPorperties.getName()); 
  12.         return strut; 
  13.     } 

五、新增配置发现文件

在resources文件夹中新建 META-INF 文件夹,在 META-INF 文件夹中新建配置发现文件 spring.factories,并且将自动配置类写到文件里。

  1. org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.javatip.str.configuration.JavatipAutoConfiguration 

六、打包测试

使用 mvn install 命令将项目打包推送到本地maven仓库,然后新建一个测试项目,引入打包好的依赖。

  1. <dependency> 
  2.     <groupId>com.javatip</groupId> 
  3.     <artifactId>javatip-spring-boot-starter</artifactId> 
  4.     <version>0.0.1-SNAPSHOT</version> 
  5. </dependency> 

在 application.yml 文件中写入自动拼接的字符串对应的属性 javatip.name 。

  1. javatip: 
  2.   name: Java旅途 

然后手写一个测试类:

  1. @RestController 
  2. public class Test { 
  3.      
  4.     @Autowired 
  5.     private StrUt strUt; 
  6.  
  7.     @GetMapping("test"
  8.     public String test(){ 
  9.  
  10.         String str = strUt.strTo("who are you?"); 
  11.         return str; 
  12.     } 

运行测试类后,页面返回了

  1. who are you?---Java旅途 

这样,一个简单的starter就写好了,只要理解了starter的原理,实现起来就很简单,第一点就是starter相当于一个依赖组,另一点就是starter可以完成初始化配置。

本文转载自微信公众号「Java旅途」,可以通过以下二维码关注。转载本文请联系Java旅途公众号。

 

责任编辑:武晓燕 来源: Java旅途
相关推荐

2021-02-03 09:04:11

解密Spring配置

2020-11-04 07:56:19

工具Linux 翻译

2021-02-22 11:13:17

VS Code代码编程

2020-09-27 10:55:10

代码Java字符串

2020-06-12 09:07:03

技术总监数据库

2021-02-03 07:56:08

版本游戏逻辑

2021-11-04 17:23:03

Java对象 immutable

2022-01-21 07:35:06

LRU缓存java

2022-02-19 22:02:21

Redisvalue元素

2020-07-13 08:40:48

同事代码

2023-07-10 09:53:59

console开发插件

2021-11-29 07:47:57

gRPCGUI客户端

2022-05-07 13:52:22

Feign 增强包K8s

2022-04-22 08:22:50

MVCCMySQLC++

2021-12-12 18:18:15

代码元宇宙Python

2020-09-27 14:13:50

Spring BootJava框架

2021-09-13 08:41:52

职场互联网自闭

2022-05-16 08:42:26

Pandasbug

2020-04-07 08:00:02

Redis缓存数据

2021-01-18 11:27:03

Istio架构云环境
点赞
收藏

51CTO技术栈公众号