hello,你好呀,我是灰小猿,一个超会写bug的程序猿!

最近在开发SpringCloud Alibaba相关的项目的时候遇到了一个比较容易踩的坑,在这里记录一下,与大家共勉。

场景是这样的:当时我要以Ncaos作为注册中心,在多服务的情况的下,使用Gateway通过服务名称来实现动态路由,但是当我通过服务名称进行请求时,一直提示“type=Service Unavailable, status=503”,但是检查yml配置和路由设置都没有问题,这就很疑惑???

 

我的使用场景:

  • SpringBoot版本为2.6.0
  • Spring Cloud版本为2021.0.1
  • Spring Cloud Alibaba版本为2021.0.1.0

Gateway服务中的yml文件:

#如下配置一定要注意空格和yml格式正确
server:
  port: 80
spring:
  application:
    name: gateway-server
  cloud:
    #注册到nacos
    nacos:
      server-addr: localhost:8848
      username: nacos
      password: nacos
    #网关设置
    gateway:
      discovery:
        locator:
          enabled: true #开启动态路由
          lower-case-service-id: true
      enabled: true # =只要加了依赖 默认开启
      routes:
        - id: login-service-route  # 这个是路由的id 保持唯一即可
          uri: http://localhost:8081   
          predicates: # 断言是给某一个路由来设定的一种匹配规则 默认不能作用在动态路由上
            - Path=/doLogin  # 匹配规则  只要你Path匹配上了/doLogin 就往 uri 转发 并且将路径带上
        - id: teacher-service-route
          uri: lb://teacher-service
          predicates:
            - Path=/**

并且Gateway和每一个服务都通过@EnableDiscoveryClient注解加入到了Nacos中,且在Nacos中可以看到。

由于我在Gateway服务中设置了过滤器,所以通过过滤器断点发现通过服务名称来进行路由时,请求进入了过滤器并且被放行,说明请求本身没有问题,但是就是请求不到对应的接口,

经过在网上找了资料发现是由于版本问题:

原因:由于springcloud从2020版本开始弃用了Ribbon,所以Alibaba在2020及之后版本的nacos中删除了Ribbon的jar包, 因此无法通过lb路由到指定微服务,所以导致出现了503情况。

从上面的配置中可以看出在实现动态路由时,Uri我使用了lb的形式,在结合上面说到的版本变更,问题原因就找到,原来Ribbon实现负载均衡的功能不能再使用了,就需要引入新的依赖。

解决方案 :只需要引入springcloud loadbalancer包即可!并且注意该依赖是需要添加在Gateway所在的服务下的,服务提供者不需要添加(亲测)

<!--客户端负载均衡loadbalancer-->
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-loadbalancer</artifactId>
</dependency>

加入如上依赖后,重新运行网关服务并重新尝试请求,测试成功!

最后提醒一下大家,开发中如果使用的是SpringCloud Alibaba 2020.0.1.0及其之后的版本,发现异常时可以优先考虑下是否是版本存在变更或需要兼容的地方!

我是灰小猿,我们下期见!

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐