springboot使用configtree读取树形文件目录中的配置

文章目录

  • 一、介绍
  • 二、演示环境
  • 三、项目演示
    • 1. 配置文件
    • 2. 导入配置
    • 3. 检测配置属性
  • 四、应用场景
  • 五、源码解析
    • 1. ConfigTreeConfigDataLocationResolver
    • 2. ConfigTreeConfigDataLoader
  • 六、总结

一、介绍

相信绝大多数使用springboot开发项目的朋友们在添加配置时,通常都是通过以下几种方式:

  • 在classpath下添加application.yml或application.properties配置文件,或通过spring.config.location指定配置文件位置。
  • 通过spring.config.additional-location指定额外的配置文件位置。
  • 通过spring.config.import导入指定位置的配置文件。

但无论通过哪种方式,其配置的形式都是通过在配置文件中通过key - value的形式添加具体配置的,且配置文件类型为yamlproperties。如下所示:

  • properties文件内容示例

    key1 = value1
    
  • yaml文件内容示例

    key1: value1
    

其中key1作为配置名,value1作为配置值。

今天给大家介绍另一种配置形式,该配置使用文件名作为配置名,文件内容作为配置值

如文件名为username的内容如下:

admin

文件名为password的内容如下:

123456

二、演示环境

本演示项目的环境如下:

  • java:1.8
  • springboot:2.4.3

三、项目演示

本项目演示的是,在指定目录中添加配置文件,并以文件名为key,文件内容为value;然后在application.yml配置文件中通过spring.config.import指定configtree将目录中的所有配置文件添加到项目的环境中,并通过placeholder${}的形式获取配置。

1. 配置文件

我们在本地文件系统中添加配置文件,其目录结构如下所示

  /etc/app/config/admin/usernamepassworddb/usernamepasswordnacos/usernamepassword

各个配置文件内容如下所示

  • /etc/app/config/admin/username

    admin
    
  • /etc/app/config/admin/password

    123456
    
  • /etc/app/config/db/username

    mysql
    
  • /etc/app/config/db/password

    123456
    
  • /etc/app/config/nacos/username

    nacos
    
  • /etc/app/config/nacos/password

    nacos
    

2. 导入配置

application.yml配置文件中添加配置spring.config.import

注意,当我们要添加以文件名为key,文件内容为value的配置文件时,必须在路径前添加前缀configtree:,且路径最后以/结尾。另外,该路径支持*通配符。

spring:config:import:- configtree:/etc/app/config/

如上所示,springboot将读取路径/etc/app/config/(包括子目录)中的所有文件,并以文件名为key文件内容为value

3. 检测配置属性

当我们按照上面示例配置时,由于在application.yml中配置的spring.config.import目录为/etc/app/config/,因此我们可以通过admin.usernameadmin.passworddb.usernamedb.passwordnacos.usernamenacos.password获取对应文件内容的值。

下面我们启动项目对其进行检验。

启动项目:

在这里插入图片描述

输出:

在这里插入图片描述

四、应用场景

看到这里,想必很多小伙伴虽然知道springboot如何通过spring.config.import + configtree来读取以文件名为key,文件内容为value的配置,但是这种配置方式使用起来并不方便,且一个文件仅对应一个配置属性,那如果需要大量配置岂不是要创建大量文件?

其实,使用该配置方式和使用application.yml方式应该是相辅相成的,两者应当配合使用。

当我们在云平台(比如docker)上运行应用程序时,有时需要读取容器提供的配置值。而我们多数情况下都是通过该容器的环境变量来获取所需的配置,但是如果我们可能会频繁修改该环境变量或该变量需要加密时,就可能暴露出它的缺点了。因为容器的环境变量是在创建镜像的时候就确定的,当我们需要修改该环境变量时就意味着已经创建的容器需要删除了。

所以我们可以通过容器挂载卷的方式,将该环境变量保存在文件中,通过挂载卷将配置文件挂载到容器中。

五、源码解析

在前面我们源码分析springboot如何创建并配置环境3 - 配置扩展属性2文章中,简单介绍过springboot通过ConfigTreeConfigDataLocationResolverStandardConfigDataLocationResolver两种配置文件位置解析器来解析配置文件的位置,然后通过ConfigTreeConfigDataLoaderStandardConfigDataLoader来加载对应配置文件中的配置内容。

因此结合本文重点,我们应主要关注ConfigTreeConfigDataLocationResolverConfigTreeConfigDataLoader是如何解析配置文件的位置并从文件中读取配置内容的。

1. ConfigTreeConfigDataLocationResolver

首先我们查看ConfigTreeConfigDataLocationResolver是如何解析出配置文件目录的,主要分两步:①判断配置的路径是否满足解析的条件,②解析配置文件的位置。

  • 判断配置的路径是否满足解析的条件

    该判断逻辑通过isResolvable()方法完成,主要判断依据就是配置的spring.config.import值是否包含configtree:前缀,如果包含,则满足条件。

    private static final String PREFIX = "configtree:";@Override
    public boolean isResolvable(ConfigDataLocationResolverContext context, ConfigDataLocation location) {return location.hasPrefix(PREFIX);
    }
    
  • 解析配置文件的位置

    该逻辑通过方法resolve()完成,其目的是根据配置的spring.config.import目录转换为该目录下文件的资源。

    @Override
    public List<ConfigTreeConfigDataResource> resolve(ConfigDataLocationResolverContext context,ConfigDataLocation location) {try {return resolve(context, location.getNonPrefixedValue(PREFIX));}catch (IOException ex) {throw new ConfigDataLocationNotFoundException(location, ex);}
    }private List<ConfigTreeConfigDataResource> resolve(ConfigDataLocationResolverContext context, String location)throws IOException {// 目录必须以“/结尾”Assert.isTrue(location.endsWith("/"),() -> String.format("Config tree location '%s' must end with '/'", location));// 如果目录不是通配符的形式,则直接根据该目录获取该目录下文件的资源集合。if (!this.resourceLoader.isPattern(location)) {return Collections.singletonList(new ConfigTreeConfigDataResource(location));}// 如果目录是通配符的形式,则对其进一步处理,获取该目录下文件的资源集合。Resource[] resources = this.resourceLoader.getResources(location, ResourceType.DIRECTORY);List<ConfigTreeConfigDataResource> resolved = new ArrayList<>(resources.length);for (Resource resource : resources) {resolved.add(new ConfigTreeConfigDataResource(resource.getFile().toPath()));}return resolved;
    }
    

2. ConfigTreeConfigDataLoader

然后我们分析ConfigTreeConfigDataLoader是如何根据配置文件资源加载其内容的。

在这里插入图片描述

在加载配置属性中,我们看到该方法主要分两步,①根据文件资源获取文件路径path,②根据文件路径获取该文件中的配置。我们在方法结束时添加断点,然后启动项目,让代码运行到断点处。如下图所示,我们发现springboot已经按照预期将各个配置文件读取成功了。

在这里插入图片描述

任意点击其中一个元素,可以看到配置文件中的内容也已经被加载了

在这里插入图片描述

最后将其封装到ConfigData对象中返回。

六、总结

  • 通过spring.config.import + configtree:前缀的方式,加载以文件名为key、文件内容为value的配置属性。
  • configtree:应以/结尾。
  • 适用于代替在云平台中读取加密的系统环境变量的场景。


纸上得来终觉浅,绝知此事要躬行。

————————————————我是万万岁,我们下期再见————————————————

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/66431.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

对约瑟夫问题的进一步思考

约瑟夫问题重述&#xff1a; 在计算机编程的算法中&#xff0c;类似问题又称为约瑟夫环 约瑟夫环&#xff1a;N个人围成一圈&#xff0c;从第一个开始报数&#xff0c;第M个将被杀掉&#xff0c;最后剩下一个&#xff0c;其余人都将被杀掉。 例如N6&#xff0c;M5&#xff0…

[Mongodb 5.0]单机启动

安装完mongodb后&#xff0c;会自动生成下面两个目录(mongod.conf中设定的)&#xff0c;用来存放日志和数据 /var/lib/mongo (数据目录) /var/log/mongodb (日志目录) 要启动一个单机版的mongodb&#xff0c;一般有两种方式&#xff1a; 第一种启动方式&#xff1a;直接使用…

Spring 使用注解开发、代理模式、AOP

使用注解开发 在Spring4之后&#xff0c;要使用注解开发&#xff0c;必须要保证AOP的包导入了 项目搭建&#xff1a; 在配置文件中导入约束&#xff0c;增加注解支持 <?xml version"1.0" encoding"UTF-8"?> <beans xmlns"http://www.spri…

双端列表 —— Deque 接口概述,使用ArrayDeque实现队列和双端队列数据结构

Deque接口简介 Deque译为双端队列&#xff0c;在双向都能作为队列来使用&#xff0c;同时可用作栈。Deque接口的方法是对称成比例的。 Deque接口继承Queue接口&#xff0c;因此具有Queue&#xff0c;Collection&#xff0c;Iterable的方法属性。 双端队列的工作原理 在常规队…

Springboot集成ip2region离线IP地名映射-修订版

title: Springboot集成ip2region离线IP地名映射 date: 2020-12-16 11:15:34 categories: springboot description: Springboot集成ip2region离线IP地名映射 1. 背景2. 集成 2.1. 步骤2.2. 样例2.3. 响应实例DataBlock2.4. 响应实例RegionAddress 3. 打开浏览器4. 源码地址&…

K8S系列一:概念入门

I. K8S概览 1.1 K8S是什么&#xff1f; K8S是Kubernetes的全称&#xff0c;官方称其是&#xff1a; Kubernetes is an open source system for managing containerized applications across multiple hosts. It provides basic mechanisms for deployment, maintenance, and …

uniapp开发小程序-分包(微信错误码:800051)

在使用uniapp开发小程序时&#xff0c;上传的时候因为文件过大&#xff0c;显示上传失败。 以下是开发过程中遇到的问题及解决方法&#xff1a; 1. 问题一&#xff1a;因为文件过大&#xff0c;显示上传失败 ①尝试过把本地使用的图片压缩到最小&#xff1b; ②把图片转换为网…

【Vue-Router】路由入门

路由&#xff08;Routing&#xff09;是指确定网站或应用程序中特定页面的方式。在Web开发中&#xff0c;路由用于根据URL的不同部分来确定应用程序中应该显示哪个内容。 构建前端项目 npm init vuelatest //或者 npm init vitelatest安装依赖和路由 npm install npm instal…

Python学习 -- 常用函数与实例详解

在Python编程中&#xff0c;数据转换是一项关键任务&#xff0c;它允许我们在不同数据类型之间自由流动&#xff0c;从而提高代码的灵活性和效率。本篇博客将深入探讨常用的数据转换函数&#xff0c;并通过实际案例为你展示如何巧妙地在不同数据类型之间转换。 数据类型转换函…

ROSpider机器人评测报告

ROSpider机器人评测报告 最近入手了一款ROSpider六足仿生机器人&#xff0c;ROSpider是一款基于ROS 操作系统开发的智能视觉六足机器人。 外观 外观上ROSpider六足机器人如同名字一样有六只机械腿&#xff0c;整体来看像一只六腿的蜘蛛。腿上的关节处用了明亮的橙黄色很是显…

大数据-玩转数据-Flink RedisSink

一、添加Redis Connector依赖 具体版本根据实际情况确定 <dependency><groupId>org.apache.flink</groupId><artifactId>flink-connector-redis_2.11</artifactId><version>1.1.5</version> </dependency>二、启动redis 参…

Java代理模式——静态代理与动态代理

代理模式 代理模式允许你为其他对象提供一个代理&#xff0c;以控制对这个对象的访问。代理模式在不改变实际对象的情况下&#xff0c;可以在访问对象时添加额外的功能。 可以理解为代理模式为被代理对象创造了一个替身&#xff0c;调用者可以通过这个替身去实现这个被代理对…