认识和安装 Nacos

Nacos 是阿里巴巴的产品,现在是 SpringCloud 中的一个组件。相比 Eureka 功能更加丰富,在国内受欢迎程度较高。

Docker 安装:官网

服务注册到 Nacos

Nacos 是 Spring Cloud Alibaba 的组件,而 Spring Cloud Alibaba 也遵循 Spring Cloud 中定义的服务注册、服务发现规范。因此使用 Nacos 和使用 Eureka 对于微服务来说,并没有太大区别。其主要差异在于:依赖不同、服务地址不同。

引入 Nacos 依赖

在 spring-cloud-demo 父工程的 pom 文件中的 <dependencyManagement> 中引入 Spring Cloud Alibaba 的依赖:

  1. <dependency>
  2. <groupId>com.alibaba.cloud</groupId>
  3. <artifactId>spring-cloud-alibaba-dependencies</artifactId>
  4. <version>2.2.6.RELEASE</version>
  5. <type>pom</type>
  6. <scope>import</scope>
  7. </dependency>

然后在 user-service 和 order-service 中的 pom 文件中引入 nacos-discovery 依赖:

<dependency>
    <groupId>com.alibaba.cloud</groupId>
    <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
</dependency>

注意:不要忘了注释掉 eureka 的依赖。

配置 Nacos 地址

在 user-service 和 order-service 的 application.yml 中添加 nacos 地址:

spring:
  cloud:
    nacos:
      server-addr: nacos:8848

这里使用的是远程服务器的地址

启动项目

启动 user-service 和 order-service 浏览器输入 http://halo:8848/nacos/ 账号密码默认都为 nacos,项目成功可以看到服务列表中的服务。
Nacos 注册中心 - 图1

服务分级存储模型

一个服务可以有多个实例,例如我们的 user-service,可以有:

  • 127.0.0.1:8081
  • 127.0.0.1:8082
  • 127.0.0.1:8083

假如这些实例分布于全国各地的不同机房,例如:

  • 127.0.0.1:8081,在上海机房
  • 127.0.0.1:8082,在上海机房
  • 127.0.0.1:8083,在杭州机房

Nacos 就将同一机房内的实例划分为一个集群

也就是说,user-service 是服务,一个服务可以包含多个集群,如杭州、上海,每个集群下可以有多个实例,形成分级模型,如图:
Nacos 注册中心 - 图2
微服务互相访问时,应该尽可能访问同集群实例,因为本地访问速度更快。当本集群内不可用时,才访问其它集群。例如:杭州机房内的 order-service 应该优先访问同机房的 user-service。

给 user-service 配置集群

修改 user-service 的 application.yml 文件,添加集群配置:

spring:
  cloud:
    nacos:
      server-addr: halo:8848
      discovery:
        cluster-name: HZ # 集群名称

重启两个 user-service 实例后,我们再次复制一个 user-service 启动配置,添加属性:

-Dserver.port=8083 -Dspring.cloud.nacos.discovery.cluster-name=SH

Nacos 注册中心 - 图3
我们可以在 Nacos 控制台看到下面结果:
Nacos 注册中心 - 图4

同集群优先的负载均衡

默认的 ZoneAvoidanceRule 并不能实现根据同集群优先来实现负载均衡。

因此 Nacos 中提供了一个 NacosRule 的实现,可以优先从同集群中挑选实例。

首先给 order-service 配置集群信息,方法同上一节

修改 order-service 的 application.yml 文件,添加集群配置:

spring:
  cloud:
    nacos:
      server-addr: halo:8848
      discovery:
        cluster-name: HZ # 集群名称

修改负载均衡规则。修改 order-service 的 application.yml 文件,修改负载均衡规则:

user-service:
  ribbon:
    NFLoadBalancerRuleClassName: com.alibaba.cloud.nacos.ribbon.NacosRule # 负载均衡规则

本地集群中采用随机

Nacos Rule 负载均衡策略:

  1. 优先选择同集群服务实例列表
  2. 本地集群找不到提供者,才去其它集群寻找,并且会报警告
  3. 确定了可用实例列表后,再采用随机负载均衡挑选实例

    权重配置

实际部署中会出现这样的场景:服务器设备性能有差异,部分实例所在机器性能较好,另一些较差,我们希望性能好的机器承担更多的用户请求。

但默认情况下 Nacos Rule 是同集群内随机挑选,不会考虑机器的性能问题。

因此,Nacos 提供了权重配置来控制访问频率,权重越大则访问频率越高。

在 Nacos 控制台,找到 user-service 的实例列表,点击编辑,即可修改权重:
Nacos 注册中心 - 图5

注意:如果权重修改为 0,则该实例永远不会被访问

环境隔离

Nacos 提供了 namespace 来实现环境隔离功能:

  • Nacos 中可以有多个 namespace
  • namespace 下可以有 group、service 等
  • 不同 namespace 之间相互隔离,例如不同 namespace 的服务互相不可见

Nacos 注册中心 - 图6

创建 namespace

默认情况下,所有 service、data、group 都在同一个 namespace,名为 public:
Nacos 注册中心 - 图7
我们可以点击页面新增按钮,添加一个 namespace

给微服务配置 namespace

给微服务配置 namespace 只能通过修改配置来实现。

例如,修改 order-service 的 application.yml 文件:

spring:
  cloud:
    nacos:
      server-addr: halo:8848
      discovery:
        cluster-name: HZ
        namespace: 45e30304-1b64-4c21-8c83-22309949af10 # 命名空间,填ID

重启 order-service 后,访问控制台,可以看到下面的结果:
Nacos 注册中心 - 图8
此时访问 order-service,因为 namespace 不同,会导致找不到 user-service,控制台会报错:

09-09 22:11:49:886 ERROR 4100 --- [nio-8080-exec-1] o.a.c.c.C.[.[.[/].[dispatcherServlet]    : Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processing failed; nested exception is java.lang.IllegalStateException: No instances available for user-service] with root cause