首页 > 其他分享 >Spring Boot读取外部配置文件失败,原因绝对出乎你意料

Spring Boot读取外部配置文件失败,原因绝对出乎你意料

时间:2024-12-03 09:11:39浏览次数:7  
标签:java 配置文件 spring Boot jar application Spring config

开心一刻

今天和相亲对象见面,特意打扮了一番
见完面回到家后我给她发微信
我:我今天的形象怎么样
她:挺白净亮眼的
我:头发不油吧
她:反光,没看清
我:???

反光没看清

知识回顾

在我们的实际开发工程中,打包的 jar 通常会包含配置文件(例如:application.yml)来作为默认配置文件,然后在不同的环境用外部配置文件来覆盖 jar 包中配置文件配置的某些配置项,当然也可以全量覆盖;Spring Boot 关于外部配置(Externalized Configuration)有这么一段说明

External Application Properties

我给大家翻译一下

应用启动时,Spring Boot会自动从以下位置查找并加载 application.propertiesapplication.yaml

  1. 从类路径

    a. 类路径根目录

    classpath根目录

    src/main/resources 下的文件,默认情况下会被打包到类路径(classpath)下

    b. 类路径下的 config

  2. 从当前目录

    a. 当前目录,也就 jar 所在目录

    b. 当前目录下的 config 目录

    c. 当前目录下的 config 目录的直接子目录

Spring Boot 会按如上顺序从上往下查找并加载 application.propertiesapplication.yaml,如果配置项重命名了,后加载的值会覆盖掉之前加载的值。配置文件中的配置项会以 PropertySources 实例的形式添加到 Spring 环境中

我们来看个示例:spring-boot-external-config,代码非常简单,我们只需要关注 ConfigDemo.java

/**
 * @author: 青石路
 */
@Component
public class ConfigDemo implements InitializingBean {

    private static final Logger LOGGER = LoggerFactory.getLogger(ConfigDemo.class);

    @Value("${retry.times}")
    private Integer retryTimes;
    @Value("${http.url}")
    private String httpUrl;

    @Override
    public void afterPropertiesSet() throws Exception {
        LOGGER.info("retryTimes:{}, httpUrl:{}", retryTimes, httpUrl);
    }
}

application.yml 内容如下

retry:
  times: 6
http:
  url: http://localhost:8080

我相信你们都能看懂,通过 Spring 注入进来两个配置项(retry.timeshttp.url)值,当应用中的全部属性都设置完成之后,Spring 会调用 afterPropertiesSet 方法,日志输出 retryTimeshttpUrl 的值。结合上述的加载顺序,我们来验证下是否如 Spring Boot 官方所说

  1. 从类路径

    这个我们只验证一种情况:类路径下只存在 application.yml;验证非常简单,直接在 jar 所在目录下

    类路径下配置文件生效

    执行

    java -jar spring-boot-external-config-1.0-SNAPSHOT.jar
    

    可以看到输出如下

    类路径下配置文件生效验证

    是不是没毛病?

  2. 从当前目录

    在 jar 所在目录下放一个 application.yml

    当前目录下配置文件

    其内容如下

    retry:
      times: 2
    

    我特意拿掉了配置项 http.url,同样直接执行

    java -jar spring-boot-external-config-1.0-SNAPSHOT.jar
    

    输出结果如下

    当前目录下配置文件生效

    红框框住的值与蓝框框住的值,它们分别来自哪个配置文件,你们应该知道吧;也如 Spring Boot 官方所说,非常正常。我们再在当前目录下加个 config 目录

    当前目录下config配置文件

    其下放一个 application.yml,内容如下

    retry:
      times: 9
    http:
      url: http://127.0.0.1:8080
    

    执行结果如下

    当前目录下config配置文件验证

    也如 Spring Boot 官方描述的那样,没有任何毛病

    为了保持部署结构的简单清晰,我们往往会采用 config 目录的这种方式来放外部配置文件,包括应用配置文件、日志配置文件等等;但如果外部配置文件跟当前目录没有直接关系了,比如在其他盘或者其他目录下,那么如何指定外部配置文件呢?有但不限于如下两种(仔细斟酌如下写法是否正确

    1. java -jar spring-boot-external-config-1.0-SNAPSHOT.jar -Dspring.config.location=外部文件路径
    2. java -jar spring-boot-external-config-1.0-SNAPSHOT.jar --spring.config.location=外部文件路径

    外部文件按路径可以是绝对路径,也可以是相对路径,如果是相对路径,则以 jar 包所在的目录开始算

复现问题

一切准备就绪,我们往 Linux 服务器上部署,先将 spring-boot-external-config-1.0-SNAPSHOT.jar 上传到服务器,执行

java -jar spring-boot-external-config-1.0-SNAPSHOT.jar

日志输出如下

linux上类路径配置

这是 jar 中默认配置文件的配置值,没问题吧?我们再上传 jar 同级目录下的 config 文件夹,上传后目录结构如下

Linux上目录结构

执行后日志输出如下

linux上config配置

也和在 windows 上的演示结果一样,很正常。绝大部分情况下,我们的配置文件不止一个,所以从命名的准确性考虑,我们往往会将 config 目录命名成 configs

linux上configs配置

那么此时启动命令就需要调整下了

java -jar spring-boot-external-config-1.0-SNAPSHOT.jar -Dspring.config.location=configs/application.yml

java -jar spring-boot-external-config-1.0-SNAPSHOT.jar --spring.config.location=configs/application.yml

我们以 -D 的形式启动下,日志输出如下

linux上configs配置-D

这输出的还是 jar 包中的配置项值,并非 configs 目录下 application.yml 中的配置项值,读取外部配置文件失败了?带着疑问我们尝试下 -- 方式

java -jar spring-boot-external-config-1.0-SNAPSHOT.jar --spring.config.location=configs/application.yml

发现能够正常加载

linux上configs配置--

为什么 -D 的方式会失败?

why

解决问题

出题出在哪,就出在 java 命令参数的顺序上,我们看下 java 命令的帮助文档

也就是说 -D-- 需要在 -jar 前面,所以正确读取 configs 目录下配置文件的命令应该是

java -Dspring.config.location=configs/application.yml -jar spring-boot-external-config-1.0-SNAPSHOT.jar

执行结果如下

正确加载外部目录的方式

总结

  1. 常用加载外部配置文件的命令参数

    java -Dspring.config.location=configs/application.yml -jar spring-boot-external-config-1.0-SNAPSHOT.jar

    java -jar spring-boot-external-config-1.0-SNAPSHOT.jar --spring.config.location=configs/application.yml

    -D 紧随 java 之后,在 -jar 之前,是 JVM 参数;-- 在 jar 名称之后,是 Spring Boot 命令行参数。两种方式有其各自的顺序写法,切勿张冠李戴!

  2. 推荐做法

    在 jar 包所在目录创建 config 文件夹,将外部配置文件置于该 config 文件夹下,启动的时候可以减少启动参数,那么出错的概率就会降低

标签:java,配置文件,spring,Boot,jar,application,Spring,config
From: https://www.cnblogs.com/youzhibing/p/18582949

相关文章

  • 如何在PbootCMS中添加自定义的清理脚本?
    在PbootCMS中添加自定义的清理脚本,可以通过扩展控制器的方法来实现。以下是详细步骤:编辑控制器文件:打开文件 /apps/home/controller/ExtLabelController.php。找到以下代码:php //测试扩展单个标签privatefunctiontest(){$this->content=str_replace('{pboot......
  • 在PbootCMS中如何使用[list:isico]变量判断文章是否有缩略图?
    在PbootCMS中,[list:isico] 变量用于判断文章是否上传了缩略图。通过这个变量,你可以在模板中灵活地控制文章列表中的图片显示逻辑。以下是详细的使用方法和示例:理解[list:isico]变量:[list:isico] 是一个布尔变量,返回值为1表示文章已上传缩略图,返回值为0表示文章未上传缩略图......
  • 如何在PbootCMS中优化图片上传和显示?
    在使用PbootCMS发布内容时,优化图片上传和显示是非常重要的,这不仅关系到网站的视觉效果,还直接影响到用户的体验和页面的加载速度。以下是一些优化图片上传和显示的方法和步骤:调整缩略图配置:打开 \config\config.php 文件,找到缩略图配置部分:php //缩略图配置'ico'=......
  • PbootCMS中istop标签不起作用,如何确保文章在列表中置顶?
     在PbootCMS中,istop标签用于标识文章是否置顶。如果发现设置istop后文章没有在列表中置顶,可能是由于前端模板调用或配置的问题。以下是详细的排查和解决方法:确认后台设置:确保在后台正确设置了文章的置顶状态。登录后台管理系统,进入“内容管理”->“文章管理”,选择文章并勾......
  • 在进行空间迁移时,如何修改 Z-BlogPHP 的配置文件?
    在进行空间迁移时,修改Z-BlogPHP的配置文件 c_option.php 是一个关键步骤,确保你的博客在新服务器上能够正常运行。以下是详细的步骤和注意事项:备份原文件:在修改配置文件之前,务必备份原文件。可以通过FTP或SFTP下载 c_option.php 文件到本地计算机,确保在修改过程中出......
  • PbootCMS模板后台编辑器无法上传图片,提示“后端配置项没有正常加载,上传插件不能正常使
    当您在使用PbootCMS模板后台编辑器时,如果遇到无法上传图片,并且提示“后端配置项没有正常加载,上传插件不能正常使用!”的问题,通常是由于后端配置项返回格式出错导致的。以下是详细的解决步骤:检查时区设置:这个问题的一个常见原因是时区设置不正确。在Linux环境下,时区设置是区分......
  • PbootCMS如何确认服务器是否支持pdo_sqlite扩展?
    在使用PbootCMS时,如果遇到“未检测到您服务器环境的sqlite3数据库扩展”的提示,您可以选择将数据库配置连接驱动改为pdo_sqlite。为了确保这种方法可行,您需要先确认服务器是否支持pdo_sqlite扩展。以下是具体的操作步骤:创建phpinfo文件:在您的网站根目录下创建一个名为info.php......
  • PbootCMS 添加栏目时报错“该内容栏目编号已经存在,不能再使用”,如何解决?
    当你在PbootCMS中尝试添加新的栏目时,如果遇到“该内容栏目编号已经存在,不能再使用”的错误提示,通常是因为数据库中的栏目编号(scode)已经存在重复值。为了解决这个问题,你可以按照以下步骤操作:备份数据库:在进行任何数据库操作之前,建议先备份整个数据库,以防止意外情况导致数据丢失......
  • springboot在线宠物用品交易网站的设计与实现(代码+数据库+LW)
    摘要随着信息技术在管理上越来越深入而广泛的应用,管理信息系统的实施在技术上已逐步成熟。本文介绍了在线宠物用品交易网站的开发全过程。通过分析在线宠物用品交易网站管理的不足,创建了一个计算机管理在线宠物用品交易网站的方案。文章介绍了在线宠物用品交易网站的系统分析......
  • springboot房屋租赁管理系统的设计与实现(代码+数据库+LW)
    摘 要互联网发展至今,无论是其理论还是技术都已经成熟,而且它广泛参与在社会中的方方面面。它让信息都可以通过网络传播,搭配信息管理工具可以很好地为人们提供服务。针对房屋租赁信息管理混乱,出错率高,信息安全性差,劳动强度大,费时费力等问题,采用房屋租赁管理系统可以有效管理,使......