微服务的配置共享

news/2025/1/13 23:27:30/

1.什么是微服务的配置共享

微服务架构中,配置共享是一个重要环节,它有助于提升服务间的协同效率和数据一致性。以下是对微服务配置共享的详细阐述:

1.1.配置共享的概念

配置共享是指在微服务架构中,将某些通用或全局的配置信息集中管理,并允许多个微服务共享这些配置。这些配置信息可能包括数据库连接信息、应用服务器设置、第三方服务接入密钥等。

1.2.配置共享的实现方式

  1. 使用配置中心:配置中心是微服务架构中常用的配置管理工具,如Nacos、Apollo、Spring Cloud Config等。这些工具提供了集中的配置管理、热更新、配置共享等功能。通过将配置信息存储在配置中心,微服务可以在启动时或运行时从配置中心获取所需的配置,从而实现配置共享。
  2. 服务间通信:在某些情况下,微服务之间可能需要通过通信来共享配置信息。例如,一个微服务可能需要将其配置信息暴露给另一个微服务以供使用。这种方式通常通过REST API、gRPC等通信协议来实现。
  3. 使用数据库:对于需要在多个微服务之间共享且频繁更新的配置信息,可以考虑将其存储在数据库中。每个微服务都可以从数据库中读取所需的配置信息,并在需要时更新数据库中的配置。这种方式需要确保数据库的一致性和可用性。

1.3.配置共享的优势

  1. 提升协同效率:通过配置共享,微服务可以更容易地协同工作,因为它们可以共享相同的配置信息,从而减少配置不一致导致的错误和冲突。
  2. 简化配置管理:配置中心等工具提供了集中的配置管理功能,使得配置信息的维护和管理变得更加简单和高效。
  3. 提高灵活性:配置共享允许微服务在运行时动态地获取和更新配置信息,从而提高了系统的灵活性和可扩展性。

1.4.配置共享的注意事项

  1. 安全性:配置信息中可能包含敏感数据,如数据库密码、API密钥等。因此,在配置共享过程中需要确保这些敏感数据的安全性,防止泄露和滥用。
  2. 一致性:配置共享需要确保多个微服务之间获取到的配置信息是一致的,以避免因配置不一致而导致的错误和冲突。
  3. 可靠性:配置中心或数据库等存储配置信息的系统需要具有高可靠性和可用性,以确保微服务在需要时能够获取到正确的配置信息。

以Nacos为例,它是一款功能丰富的开源平台,提供了强大的配置管理能力。在微服务架构中,可以使用Nacos作为配置中心来实现配置共享。通过Nacos的配置中心功能,可以将所有的配置文件集中管理,每个服务从Nacos的配置中心获取自己的配置。这样,就可以在一个地方统一管理所有的配置,大大简化了配置管理的复杂性。同时,Nacos还支持配置的热更新功能,当配置信息发生变化时,可以自动将新的配置推送给相关的微服务,而无需重启服务。

综上所述,微服务配置共享是提升服务间协同效率和数据一致性的重要手段。通过选择合适的实现方式并注意相关的安全性、一致性和可靠性问题,可以有效地实现微服务之间的配置共享。

微服务共享的配置可以统一交给Nacos保存和管理,在Nacos控制台修改配置后,Nacos会将配置变更推送给相关的微服务,并且无需重启即可生效,实现配置热更新。

网关的路由同样是配置,因此同样可以基于这个功能实现动态路由功能,无需重启网关即可修改路由配置。

2.配置共享

我们可以把微服务共享的配置抽取到Nacos中统一管理,这样就不需要每个微服务都重复配置了。分为两步:

  • 在Nacos中添加共享配置

  • 微服务拉取配置

就是在多个服务中相似的配置提取出来,给各个服务使用,只需要修改自己的部分配置即可,

比如,每个微服务可以有自己的数据库,这必然需要JDBC数据库连接,这时候只需要改自己的数据库名字即可使用。

jdbc相关配置,在配置管理->配置列表中点击+新建一个配置:

如果没有NACOS可以去看

服务注册和发现-CSDN博客icon-default.png?t=O83Ahttps://blog.csdn.net/2402_85591461/article/details/144810700?spm=1001.2014.3001.5501这篇博客,教你如何配置NACOS

在弹出的表单中填写信息:

注意这里的jdbc的相关参数并没有写死,例如:

  • 数据库ip:通过${hm.db.host:192.168.150.101}配置了默认值为192.168.150.101,同时允许通过${hm.db.host}来覆盖默认值

  • 数据库端口:通过${hm.db.port:3306}配置了默认值为3306,同时允许通过${hm.db.port}来覆盖默认值

  • 数据库database:可以通过${hm.db.database}来设定,无默认值

3.拉取共享配置

接下来,我们要在微服务拉取共享配置。将拉取到的共享配置与本地的application.yaml配置合并,完成项目上下文的初始化。

不过,需要注意的是,读取Nacos配置是SpringCloud上下文(ApplicationContext)初始化时处理的,发生在项目的引导阶段。然后才会初始化SpringBoot上下文,去读取application.yaml

也就是说引导阶段,application.yaml文件尚未读取,根本不知道nacos 地址,该如何去加载nacos中的配置文件呢?

SpringCloud在初始化上下文的时候会先读取一个名为bootstrap.yaml(或者bootstrap.properties)的文件,如果我们将nacos地址配置到bootstrap.yaml中,那么在项目引导阶段就可以读取nacos中的配置了。

因此,微服务整合Nacos配置管理的步骤如下:

1)引入依赖:

  <!--nacos配置管理--><dependency><groupId>com.alibaba.cloud</groupId><artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId></dependency><!--读取bootstrap文件--><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-bootstrap</artifactId></dependency>

2)新建bootstrap.yaml

内容如下:

spring:application:name: cart-service # 服务名称profiles:active: devcloud:nacos:server-addr: 192.168.150.101 # nacos地址config:file-extension: yaml # 文件后缀名shared-configs: # 共享配置- dataId: shared-jdbc.yaml # 共享mybatis配置- dataId: shared-log.yaml # 共享日志配置- dataId: shared-swagger.yaml # 共享日志配置

3)修改application.yaml

由于一些配置挪到了bootstrap.yaml,因此application.yaml需要修改为:

server:port: 8082
feign:okhttp:enabled: true # 开启OKHttp连接池支持
hm:swagger:title: 购物车服务接口文档package: com.hmall.cart.controllerdb:database: hm-cart

重启服务,发现所有配置都生效了。

4.配置热更新

配置热更新是软件开发中的一个重要技术手段,它允许开发者在不关闭应用程序或服务的情况下,对应用程序的配置信息进行即时更新。以下是对配置热更新的详细解释:

4.1.定义

配置热更新是指在外部配置中心(如Nacos)的配置项发生变化时,应用端能够自动同步这些最新的配置数据,而无需重启应用。这种技术使得应用程序能够即时响应配置的变化,从而提高系统的灵活性和可用性。

Nacos的配置热更新能力了,分为两步:

  • 在Nacos中添加配置

  • 微服务读取配置

4.2.添加配置到Nacos

首先,我们在nacos中添加一个配置文件

注意文件的dataId格式:

[服务名]-[spring.active.profile].[后缀名]

文件名称由三部分组成:

  • 服务名:我们是购物车服务,所以是cart-service

  • spring.active.profile:就是spring boot中的spring.active.profile,可以省略,则所有profile共享该配置

  • 后缀名:例如yaml

这里我们直接使用cart-service.yaml这个名称,则不管是dev还是local环境都可以共享该配置。

配置内容如下:

hm:cart:maxAmount: 1 # 购物车商品数量上限

提交配置,在控制台能看到新添加的配置:

4.3.配置热更新

接着,我们在微服务中读取配置,实现配置热更新。

服务中新建一个属性读取类:

代码如下

package com.hmall.cart.config;import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;@Data
@Component
@ConfigurationProperties(prefix = "hm.cart")
public class CartProperties {private Integer maxAmount;
}

接着,在业务中使用该属性加载类:

加入成功!

无需重启服务,配置热更新就生效了!

5.动态路由

网关的路由配置全部是在项目启动时由org.springframework.cloud.gateway.route.CompositeRouteDefinitionLocator在项目启动的时候加载,并且一经加载就会缓存到内存中的路由表内(一个Map),不会改变。也不会监听路由变更,所以,我们无法利用上节课学习的配置热更新来实现路由更新。

因此,我们必须监听Nacos的配置变更,然后手动把最新的路由更新到路由表中。这里有两个难点:

  • 如何监听Nacos配置变更?

  • 如何把路由信息更新到路由表?

5.1.监听Nacos配置变更

在Nacos官网中给出了手动监听Nacos配置变更的SDK:

Java SDKicon-default.png?t=O83Ahttps://nacos.io/zh-cn/docs/sdk.html

这里核心的步骤有2步:

  • 创建ConfigService,目的是连接到Nacos

  • 添加配置监听器,编写配置变更的通知处理逻辑

由于我们采用了spring-cloud-starter-alibaba-nacos-config自动装配,因此ConfigService已经在com.alibaba.cloud.nacos.NacosConfigAutoConfiguration中自动创建好了:

NacosConfigManager中是负责管理Nacos的ConfigService的,具体代码如下:

因此,只要我们拿到NacosConfigManager就等于拿到了ConfigService,第一步就实现了。

第二步,编写监听器。虽然官方提供的SDK是ConfigService中的addListener,不过项目第一次启动时不仅仅需要添加监听器,也需要读取配置,因此建议使用的API是这个:

String getConfigAndSignListener(String dataId, // 配置文件idString group, // 配置组,走默认long timeoutMs, // 读取配置的超时时间Listener listener // 监听器
) throws NacosException;

既可以配置监听器,并且会根据dataId和group读取配置并返回。我们就可以在项目启动时先更新一次路由,后续随着配置变更通知到监听器,完成路由更新。

5.2.更新路由

更新路由要用到org.springframework.cloud.gateway.route.RouteDefinitionWriter这个接口:

package org.springframework.cloud.gateway.route;import reactor.core.publisher.Mono;/*** @author Spencer Gibb*/
public interface RouteDefinitionWriter {/*** 更新路由到路由表,如果路由id重复,则会覆盖旧的路由*/Mono<Void> save(Mono<RouteDefinition> route);/*** 根据路由id删除某个路由*/Mono<Void> delete(Mono<String> routeId);}

这里更新的路由,也就是RouteDefinition,之前我们见过,包含下列常见字段:

  • id:路由id

  • predicates:路由匹配规则

  • filters:路由过滤器

  • uri:路由目的地

将来我们保存到Nacos的配置也要符合这个对象结构,将来我们以JSON来保存,格式如下:

{"id": "item","predicates": [{"name": "Path","args": {"_genkey_0":"/items/**", "_genkey_1":"/search/**"}}],"filters": [],"uri": "lb://item-service"
}

以上JSON配置就等同于:

spring:cloud:gateway:routes:- id: itemuri: lb://item-servicepredicates:- Path=/items/**,/search/**

OK,我们所需要用到的SDK已经齐全了。

5.3.实现动态路由

首先, 我们在网关gateway引入依赖:

<!--统一配置管理-->
<dependency><groupId>com.alibaba.cloud</groupId><artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId>
</dependency>
<!--加载bootstrap-->
<dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-bootstrap</artifactId>
</dependency>

然后在网关gatewayresources目录创建bootstrap.yaml文件,内容如下:

spring:application:name: gatewaycloud:nacos:server-addr: 192.168.150.101config:file-extension: yamlshared-configs:- dataId: shared-log.yaml # 共享日志配置

接着,修改gatewayresources目录下的application.yml,把之前的路由移除,最终内容如下:

server:port: 8080 # 端口
hm:jwt:location: classpath:hmall.jks # 秘钥地址alias: hmall # 秘钥别名password: hmall123 # 秘钥文件密码tokenTTL: 30m # 登录有效期auth:excludePaths: # 无需登录校验的路径- /search/**- /users/login- /items/**

然后,在gateway中定义配置监听器:

其代码如下:

package com.hmall.gateway.route;import cn.hutool.json.JSONUtil;
import com.alibaba.cloud.nacos.NacosConfigManager;
import com.alibaba.nacos.api.config.listener.Listener;
import com.alibaba.nacos.api.exception.NacosException;
import com.hmall.common.utils.CollUtils;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.cloud.gateway.route.RouteDefinition;
import org.springframework.cloud.gateway.route.RouteDefinitionWriter;
import org.springframework.stereotype.Component;
import reactor.core.publisher.Mono;import javax.annotation.PostConstruct;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.concurrent.Executor;@Slf4j
@Component
@RequiredArgsConstructor
public class DynamicRouteLoader {private final RouteDefinitionWriter writer;private final NacosConfigManager nacosConfigManager;// 路由配置文件的id和分组private final String dataId = "gateway-routes.json";private final String group = "DEFAULT_GROUP";// 保存更新过的路由idprivate final Set<String> routeIds = new HashSet<>();@PostConstructpublic void initRouteConfigListener() throws NacosException {// 1.注册监听器并首次拉取配置String configInfo = nacosConfigManager.getConfigService().getConfigAndSignListener(dataId, group, 5000, new Listener() {@Overridepublic Executor getExecutor() {return null;}@Overridepublic void receiveConfigInfo(String configInfo) {updateConfigInfo(configInfo);}});// 2.首次启动时,更新一次配置updateConfigInfo(configInfo);}private void updateConfigInfo(String configInfo) {log.debug("监听到路由配置变更,{}", configInfo);// 1.反序列化List<RouteDefinition> routeDefinitions = JSONUtil.toList(configInfo, RouteDefinition.class);// 2.更新前先清空旧路由// 2.1.清除旧路由for (String routeId : routeIds) {writer.delete(Mono.just(routeId)).subscribe();}routeIds.clear();// 2.2.判断是否有新的路由要更新if (CollUtils.isEmpty(routeDefinitions)) {// 无新路由配置,直接结束return;}// 3.更新路由routeDefinitions.forEach(routeDefinition -> {// 3.1.更新路由writer.save(Mono.just(routeDefinition)).subscribe();// 3.2.记录路由id,方便将来删除routeIds.add(routeDefinition.getId());});}
}

接下来,我们直接在Nacos控制台添加路由,路由文件名为gateway-routes.json,类型为json

[{"id": "item","predicates": [{"name": "Path","args": {"_genkey_0":"/items/**", "_genkey_1":"/search/**"}}],"filters": [],"uri": "lb://item-service"},{"id": "cart","predicates": [{"name": "Path","args": {"_genkey_0":"/carts/**"}}],"filters": [],"uri": "lb://cart-service"},{"id": "user","predicates": [{"name": "Path","args": {"_genkey_0":"/users/**", "_genkey_1":"/addresses/**"}}],"filters": [],"uri": "lb://user-service"},{"id": "trade","predicates": [{"name": "Path","args": {"_genkey_0":"/orders/**"}}],"filters": [],"uri": "lb://trade-service"},{"id": "pay","predicates": [{"name": "Path","args": {"_genkey_0":"/pay-orders/**"}}],"filters": [],"uri": "lb://pay-service"}
]

无需重启网关,稍等几秒钟后,网关路由成功了!


http://www.ppmy.cn/news/1562904.html

相关文章

如何解决服务器中 MySQL 的死锁问题

解决MySQL中的死锁问题是数据库管理中的一个重要课题&#xff0c;尤其是在高并发的环境下。死锁是指两个或多个事务在执行过程中&#xff0c;相互等待对方释放持有的锁&#xff0c;从而导致所有相关事务无法继续执行。为了解决这一问题&#xff0c;可以采取以下几种方法和最佳实…

Git分支——《Pro Git》

分支简介 分支的重要性 作用&#xff1a;分支允许将工作从主开发线上分离&#xff0c;避免影响主线开发。传统版本控制的劣势&#xff1a;创建分支通常需要复制整个项目文件&#xff0c;效率低下。Git 的优势&#xff1a;分支是 Git 的“必杀技特性”&#xff0c;创建和切换分…

10_Redis数据结构-HyperLogLog基数统计

1.HyperLogLog介绍 1.1 基本概念 Redis中的HyperLogLog是一种用于估算集合中不同元素数量(即基数)的概率数据结构。它特别适用于处理非常大的数据集,与传统的精确计数方法相比,HyperLogLog可以在消耗极少量内存的情况下提供相对准确的估计值。HyperLogLog是在Redis 2.8.9…

Centos集群同步文件脚本xsync

第一次使用需要安装rsync yum -y install rsync 创建执行文件 vi /usr/local/bin/xsync 执行权限 chmod ax /usr/local/bin/xsync #!/bin/bash# 获取输出参数&#xff0c;如果没有参数则直接返回 pcount$# if [ $pcount -eq 0 ] thenecho "no parameter find !";exit…

宝塔面板 php8.0 安装 fileinfo 拓展失败

系统&#xff1a;Albaba Cloud Linux release 3 &#xff08;OpenAnolis Editon&#xff09;即 Centos 平替 异常提示&#xff1a; cc: fatal error: ** signal terminated program cc1 compilation terminated. make: *** [Makefile:211: libmagic/apprentice.lo] Error 1搜…

C# 使用iText 编辑PDF

NetCore 创建、编辑PDF插入表格、图片、文字 NetCore 创建、编辑PDF插入表格、图片、文字(二) NetCore 创建、编辑PDF插入表格、图片、文字(三) 1&#xff0c;.net8 环境&#xff0c;引入 包 itext7 itext7.bouncy-castle-adapter 2,直接上代码 public class PDFEditor{public…

MySQL 16 章——变量、流程控制和游标

一、变量 在MySQL数据库的存储过程和存储函数中&#xff0c;可以使用变量来存储查询或计算的中间结果数据&#xff0c;或者输出最终的结果数据 在MySQL数据库中&#xff0c;变量分为系统变量和用户自定义变量 &#xff08;1&#xff09;系统变量 1.1.1系统变量分类 变量由…

同域名前后端分离项目 nginx配置实践

新项目采用前后端分离的方式开发&#xff0c;前后端代码打算分开部署&#xff08;同机器且同域名&#xff09;&#xff0c;但打算支持后端依然可访问静态资源&#xff08;nginx配置仅一份&#xff09;。 搜索nginx配置大部分都通过url前缀进行转发来做前后端分离&#xff0c;不…