2021-11-22 spring-cloud-nacos配置优

2021-11-22  本文已影响0人  FredWorks

最近有项目组同学问到为什么自己配置了nacos,但配置不生效?我简单看了下,发现问题出在相关配置的优先级模式不同。
spring-boot项目,有bootstrap、application两个配置文件,结合profile,和支持的文件格式properties、yaml,已经有6个配置文件了。然后使用了spring-cloud-starter-alibaba-nacos-config 后,又提供了三级配置。这些配置之间的组合关系,将在无形中影响配置的效果。很多同学只知道其中的一种,因此在无意识引入两种或以上的配置后,就会发现有奇怪的配置不生效问题发生。

1. bootstrap.yaml

spring-boot项目依赖bootstrap.yml 用于应用程序上下文的引导阶段,由父Spring ApplicationContext加载,其工作的阶段为父ApplicationContext 被加载到使用application.yml的之前。也就是说 bootstrap 加载优先于 applicaton。

bootstrap 主要用于从额外的资源来加载配置信息,还可以在本地外部配置文件中解密属性。这两个上下文共用一个环境,它是任何Spring应用程序的外部属性的来源。bootstrap 里面的属性会优先加载,它们默认也不能被本地相同配置覆盖。

bootstrap 配置文件有以下几个应用场景:

由于spring-boot支持多种文件格式,所以多种格式之间,其优先级是平等的,只要找到了一个,就会被使用。一般有:.properties、.yaml、.xml等格式。

2. application.yaml

应用级别的spring-boot配置文件,主要用于 Spring Boot 项目的自动化配置,其加载优先级低于bootstrap.yaml。

  • 按优先级,假设当前运行目录为/,优先级依次为:/config > / > classpath:/config > classpath:/
  • 同时出现在上述四个配置文件中的同名属性,按优先级的覆盖低优先级的。
  • 分别出现在不同配置文件中的非同名属性,合并后生效。
  • 对同名但不同后缀名的配置文件,properties > yaml

可以在启动命令时,通过 --spring.config.location 命令参数指定用逗号分割的外部配置文件路径,该路径下配置高于一切。

3、spring-cloud-nacos引入的三级配置文件

nacos作为外部配置服务器,通过spring-boot的bootstrap.yaml引入。但nacos本身,也提供了三级配置体系:主配置(只有一个,但会按照不同后缀名,去找到相关配置)、扩展配置、共享配置。

三级配置的优先级如下:主配置 > 扩展配置 > 共享配置

3.1 主配置

nacos提供的配置路径 spring.cloud.nacos.config 下,有一系列的属性用于定位主配置。基于 prefix(默认为 ${spring.application.name} 的值)、namespace、group(默认为字符串 DEFAULT_GROUP)、file-extension(默认为字符串 .properties),按组装规则 ${prefix}-${spring.profiles.active}.${file-extension}去找到一个配置。

spring:
  application:
    name: ddd-demo-service
  cloud:
    nacos:
      config:
        server-addr: nacos-2.nacos-headless.public.svc.cluster.local:8848
        namespace: ygjpro-test2

上述配置,意味spring-boot和nacos 将按照如下规则执行:

  1. 按照规则 ${prefix}-${spring.profiles.active}.${file-extension}来获得dataId:ddd-demo-service.properties。因为:
  • ${prefix}:没有指定 ${prefix} ,取默认值 ${spring.application.name},为字符串 ddd-demo-service
  • spring.profiles.active:没有指定 spring.profiles.active,取默认值空。
  • ${file-extension}:没有指定 ${file-extension},取默认值字符串.properties
  1. spring-boot将尝试去nacos的 ygjpro-test2 表空间,在 DEFAULT_GROUP group 分组下,加载一个dataId叫做 ddd-demo-service 的配置。
  2. 如果发现上述dataId不存在,则继续尝试加载名为 ddd-demo-service 的dataId,该dataId只是前面步骤1中获得的dataId,去掉file-extension后缀名。
  3. 如果上述两个步骤都没有找到dataId,就不再尝试去找主配置了。

在nacos的所有配置中,主配置(存在的情况下)具有最高的优先级,其同名配置值不能被扩展配置或共享配置中定义的同名属性所覆盖。

3.2 共享配置和扩展配置

上述两类配置都支持三个属性:data-idgroup(默认为字符串 DEFAULT_GROUP)、refresh(默认为true)。

3.2.1 共享配置和扩展配置的区别

实际上,nacos中并未对 extension-configsshared-configs 的差别进行详细阐述。我们从他们的结构,看不出本质差别;除了优先级不同以外,也没有其他差别。那么,nacos项目组为什么要引入两个类似的配置呢?我们可以从当初该功能的需求(issue)上找到其原始目的。
摘要其核心内容如下:

根据nacos的默认理念:

  • namespace区分环境:开发环境、测试环境、预发布环境、生产环境
  • group区分不同应用:同一个环境内,不同应用的配置,通过group来区分。

因此,主配置应当在dataId上要区分,同时最好还要有 group 的区分,因为group区分应用(虽然dataId上区分了,不用设置group也能按应用单独加载)。

因此按该理念,shared-configs 指定的配置,本来应该是不指定 group的,也就是应当归入 DEFAULT_GROUP 这个公共分组。

比如,其他应用的数据库url,都是一个固定的url,使用 shared-configs.dataId = mysql 的共享配置。
但其中有一个应用 ddd-demo 是特例,需要为该应用配置扩展属性来覆盖。所以定义如下扩展配置:

spring:
 application:
   name: ddd-demo-service
 cloud:
   nacos:
     config:
       server-addr: nacos-2.nacos-headless.public.svc.cluster.local:8848
       namespace: ygjpro-test2
       group: ddd-demo
       ......
       shared-configs[3]:
         data-id: mysql.yaml
         refresh: true
       ......
       extension-configs[3]:
         data-id: mysql.yaml
         group: ddd-demo
         refresh: true
  • step1: 我们在group 为 DEFAULT_GROP 的分组内,创建各应用的共享配置dataId为mysql
  • step2:我们在group 为 ddd-demo 的分组内,,创建 ddd-demo 应用专有的配置mysql,其中定义的属性,会覆盖上述共享配置中定义的同名属性。

3.2.2 关于优先级

比如:

  • 同为扩展配置,存在如下优先级关系:extension-configs[3] > extension-configs[2] > extension-configs[1] > extension-configs[0]
  • 同为共享配置,存在如下优先级关系:shared-configs[3] > shared-configs[2] > shared-configs[1] > shared-configs[0]

3.2.3 在nacos中配置的注意事项

主配置有 ${file-extension} 来指定文件后缀,因此在根据规则生成 dataId时,spring-boot 知道如何去识别这个读取下来的文件。所以在nacos中配置时,dataId可以带文件后缀,也可以不要带文件后缀。
spring-boot 发现用不带文件后缀的dataId读取不到时,会尝试去掉文件后缀后的dataId 再去读取一次。

共享配置和扩展配置没有 filez-extension 属性,因此 spring-boot 会认为 dataId 本身已经包含了 文件后缀了。如果当发现按照配置中指定的dataId去nacos取回来的文件没有后缀名时,spring-boot将不会识别读取回来的文件。

上一篇 下一篇

猜你喜欢

热点阅读