Spring Boot + Docker:实现可挂载可热更新的 config.json

文章目录

前言:在实际项目中,我们经常需要让前端读取一个 config.json,并且希望它 支持在 Docker 容器里挂载覆盖、实时生效 ,避免每次修改都要重新打包。


一、背景:为什么 jar 内的 config.json 不能被覆盖?

很多人遇到这样的场景:

  • 前端请求 GET /config.json
  • 想通过 Docker 挂载一份外部 config.json 到容器里覆盖 jar 里的那份
  • 结果发现改了也不生效 → 还是读到打包进 jar 的老版本

🔎 根本原因:

Spring Boot 不会自动替换 jar 内部的静态资源文件,挂载的静态资源只是在查找路径里排在前面而已。

换句话说:

文件如果在 jar 里,通过 classpath 加载的优先级并不因为挂载而被覆盖(实际查找路径可能不同)。


二、Spring Boot 静态资源加载机制

在 Spring Boot 里,静态资源是这样被查找的:

properties 复制代码
# Spring Boot 2.4+ 使用这个
spring.web.resources.static-locations=file:/app/static/, classpath:/static/

# Spring Boot 2.3- 使用这个
spring.resources.static-locations=file:/app/static/, classpath:/static/

这个配置表示:

优先级 资源位置
第一 外部目录 file:/app/static/
第二 jar 内 classpath:/static/

注意:这是查找顺序,不是覆盖机制。


三、解决方案结构

为了做到:

✔ Docker 容器挂载的 config.json 可覆盖

✔ 修改后立即生效(无重启,无缓存)

✔ 支持 /config.json/wvp/config.json 两种访问路径

我们的整体方案是:

复制代码
前端请求 /config.json 或 /wvp/config.json
→ 后端 Controller 先尝试读取 容器内 /app/config/config.json
→ 如果存在直接返回
→ 如果不存在再 fallback 到 classpath:/static/config.json

同时设置返回头 Cache-Control: no-store 避免浏览器缓存旧配置。


四、核心实现:优先读外部文件

在你的 Controller 里写如下逻辑:

java 复制代码
@RestController
public class WvpIndexController {

    private static final String EXTERNAL_PATH = "/app/config/config.json";
    private static final String CLASSPATH_PATH = "classpath:/static/config.json";

    @GetMapping({"/config.json", "/wvp/config.json"})
    public ResponseEntity<Resource> getConfig(HttpServletResponse response) {

        Resource resource;

        // 1) 优先尝试读取外部挂载文件
        File external = new File(EXTERNAL_PATH);
        if (external.exists() && external.isFile()) {
            resource = new FileSystemResource(external);
        } else {
            // 2) 不存在则 fallback 到内置 classpath
            resource = new ClassPathResource("static/config.json");
        }

        // 设置不缓存
        response.setHeader("Cache-Control", "no-store");

        return ResponseEntity.ok()
                .contentType(MediaType.APPLICATION_JSON)
                .body(resource);
    }
}

五、完整 Dockerfile 示例

dockerfile 复制代码
FROM openjdk:17-jdk-alpine

WORKDIR /app

COPY wvp.jar /app/wvp.jar
COPY static /app/static  # 静态资源 fallback

EXPOSE 8080

ENTRYPOINT ["java", \
  "-Dspring.web.resources.static-locations=file:/app/static/,classpath:/static/", \
  "-Dspring.resources.static-locations=file:/app/static/,classpath:/static/", \
  "-jar", "wvp.jar"]

八、总结

方案 是否可挂载覆盖 是否热更新 是否兼容多个路径
只依赖 Spring Boot 静态资源
加上后端优先逻辑

✨ 推荐方案:后端优先读取外部挂载文件 + fallback classpath + no-cache

这样你只要:

bash 复制代码
docker-compose up -d

然后在宿主机改 config.json,前端刷新立即生效 💪

相关推荐
工具罗某人19 小时前
docker compose部署kafka集群搭建
docker·容器·kafka
还在忙碌的吴小二19 小时前
Harness 最佳实践:Java Spring Boot 项目落地 OpenSpec + Claude Code
java·开发语言·spring boot·后端·spring
ai产品老杨20 小时前
异构计算时代的视频底座:基于 ZLMediaKit 与 Spring Boot 的 X86/ARM 跨平台架构解析
arm开发·spring boot·音视频
老神在在00121 小时前
Spring Boot 全局异常处理器(GlobalExceptionHandler)
spring boot·spring·java-ee·状态模式·
han_hanker21 小时前
@GetMapping @PostMapping @DeleteMapping @PutMapping
spring boot
han_hanker1 天前
@Validated @Valid 用法
java·spring boot
言慢行善1 天前
SpringBoot中的注解介绍
java·spring boot·后端
许杰小刀1 天前
MyBatis-Plus实战:Spring Boot数据库操作效率提升10倍
数据库·spring boot·mybatis
sbjdhjd1 天前
Docker | 核心概念科普 + 保姆级部署
linux·运维·服务器·docker·云原生·面试·eureka
摇滚侠1 天前
Vmvare 虚拟机安装 Linux CentOS 7 操作系统 一键安装 Docker 1Panel 一键安装 MySQL Redis OpenClaw
linux·docker·centos