SpringCloud-Gateway动态路由之Nacos

本文介绍了如何在Spring Cloud Gateway中实现动态路由,避免因服务上线或下线需要重启网关。通过Nacos作为数据源,动态加载和更新路由信息,实现高可用。文章详细讲解了实现动态路由的关键点,包括自定义RouteDefinitionRepository、配置Nacos监听器以及配置类的设计。最后,作者提供了测试步骤和配置示例,展示了动态路由的实时更新效果。

原创作者:老顾
转载请注明出处

前言

当我们的网关Gateway程序开发完成之后,需要部署到生产环境,这个时候你的程序不能是单点运行的,肯定是多节点启动(独立部署或者docker等容器部署),防止单节点故障导致整个服务不能访问,网关是对客户端的入口与出口,在生产运行中极为重要,哪怕是简单的重启也会导致部分请求的丢失。

网关的路由配置这个时候就是一个大问题,是代码里面编写还是配置文件配置?他们都有一个致命的缺点,当有新的程序需要接入到网关进行路由或者有服务需要下线时候需要修改代码或者配置,然后重启整个网关程序,导致其他正常的服务路由受到了影响。

所以在实际生产环境中为了保证高可靠和高可用,是尽量要避免重启网关,所以实现动态路由是非常有必要的;

本文主要介绍Spring Cloud Gateway 实现的思路,并且把路由信息存放在外部源中,这里以Nacos为数据源来讲解。

实现要点

要实现动态路由只需关注下面4个点

  1. 网关启动时,动态路由的数据怎样加载进来
  2. 静态路由与动态路由以那个为准
  3. 监听动态路由的数据源变化
  4. 数据有变化时怎样通知gateway刷新路由

ps:静态路由指的是配置文件里写死的路由配置

具体实现

Spring Cloud Gateway 中加载路由信息分别由以下几个类负责:

  1. PropertiesRouteDefinitionLocator:从配置文件中读取路由信息(如YML、Properties等)
  2. RouteDefinitionRepository:从存储器中读取路由信息(如内存、配置中心、Redis、MySQL等)
  3. DiscoveryClientRouteDefinitionLocator:从注册中心中读取路由信息(如Nacos、Eurka、Zookeeper等)

我们可以通过自定义RouteDefinitionRepository的实现类来实现动态路由的目的。

实现动态路

有的数据加载我们可以查看源码RouteDefinitionRepository的存储的实现类,只有InMemoryRouteDefinitionRepository,此实现类是存放在内存中的。

我们可以重新定义一个nacos作为存储的实现,看下面的代码

上面代码核心的是重写 getRouteDefinitions 方法实现路由信息的读取;这个方法中涉及到了ConfigService对象Api的方法

管理Api方法,这个老顾这里不详细说了;小伙伴们可以理解为就是操作nacos config的对象

配置Nacos监听器,监听路由配置信息的变化,也是利用api方法,也就是addListener方法;此方法一看就知道,就是用来监听config信息变化的。

此addListener方法中路由变化只需要往 ApplicationEventPublisher 推送一个 RefreshRoutesEvent 事件,即刻gateway会自动监听该事件并调用 getRouteDefinitions 方法更新路由信息。

这样就达到了动态更新路由了。

配置类

为了更好的把此动态路由作为公共core,我们需要利用配置类,达到是否启用的方式;可以让开发者自行配置。

上面的配置类,有两个重要的**@ConditionalOnProperty,这个就是用来控制是否启用动态路由,以及是否用nacos作为存储。**

里面还有@Value注解定义我们存储路由信息的DataId和Group,application.yml如果不配置默认值为scg-routes和SCG_GATEWAY

添加Nacos路由配置

在同一个namespace中创建scg-routes和SCG_GATEWAY

格式选择json,json体里面的格式其实就是RouteDefinition类的属性

注意上面的配置的json是数组格式哦,也就是可以有很多路由哦

[{
    "id": "baidu",
    "order": 0,
    "predicates": [{
        "args": {
            "pattern": "/baidu/**"
        },        "name": "Path"
    }],    "uri": "https://www.baidu.com"
},{    "id": "sina",
    "order": 2,
    "predicates": [{
        "args": {
            "pattern": "/sina/**"
        },        "name": "Path"
    }],    "uri": "http://www.sina.com.cn"
}]

测试

在网关项目中的application.yml中配置

rainbow: 
  gateway: 
    dynamicRoute: 
      enabled: true

启动网关,就可以了;为了方便看到网关当前的路由信息,我们需要引入

<dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-actuator</artifactId>
</dependency>

并且在application.yml中配置

management:
  endpoints:
    web:
      exposure:
        include: '*'
  endpoint:
    gateway:
      enabled: true
    health:
      show-details: always

这样我们就可以访问http://localhost:8081/actuator/gateway/routes,就可以看到路由信息

小伙伴们可以尝试修改一下nacos中的dataId为scg-routes,我们可以发现路由及时 得到了更改。

老顾只修改了route_id的名称,路由就及时更新了

我们也看到了控制台日志也有相应的变化

总结

总结

上述知识点,囊括了目前互联网企业的主流应用技术以及能让你成为“香饽饽”的高级架构知识,每个笔记里面几乎都带有实战内容。

很多人担心学了容易忘,这里教你一个方法,那就是重复学习。

打个比方,假如你正在学习 spring 注解,突然发现了一个注解@Aspect,不知道干什么用的,你可能会去查看源码或者通过博客学习,花了半小时终于弄懂了,下次又看到@Aspect 了,你有点郁闷了,上次好像在哪哪哪学习,你快速打开网页花了五分钟又学会了。

资料领取方式:戳这里免费领取

从半小时和五分钟的对比中可以发现多学一次就离真正掌握知识又近了一步。

人的本性就是容易遗忘,只有不断加深印象、重复学习才能真正掌握,所以很多书我都是推荐大家多看几遍。哪有那么多天才,他只是比你多看了几遍书。
1R2dB)**

从半小时和五分钟的对比中可以发现多学一次就离真正掌握知识又近了一步。

[外链图片转存中…(img-SwA2Xpre-1623585643172)]

人的本性就是容易遗忘,只有不断加深印象、重复学习才能真正掌握,所以很多书我都是推荐大家多看几遍。哪有那么多天才,他只是比你多看了几遍书。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值