【微服务】(3) 服务注册与发现

一、问题出现

订单服务调用商品服务的接口时,写死 商品服务接口的url ,使用 spring 提供的 RestTemplate 进行 http 通信。这存在一个问题,每次 url 变更,都需要修改原有代码,频繁部署,十分麻烦。

二、注册中心

注册中心能解决上述问题:

  • 服务注册 :服务提供者启动时,主动把自己的接口地址注册 到注册中心,并定期发送心跳 汇报存活状态,一旦注册中心与服务提供者长时间无法通信 ,则注销该实例。
  • 服务发现 :服务消费者主动到注册中心查询服务提供者的接口地址,并通过该地址调用服务提供者的接口。

三、CAP 理论

CAP 理论是注册中心设计的基础理论。

  • 一致性Consistency):所有服务同时向注册中心查询/注册,得到的响应一致。
  • 可用性Availability):所有服务向注册中心注册/查询,都能得到响应(但响应结果可能是错的)。
  • 分区容错性Partition Tolerance):当注册中心集群中存在通信中断(网络分区),注册中心仍能继续对外提供服务。

网络故障是不可避免的,注册中心集群又不能同时全部故障,因此分区容错性是必须 的。而 一致性 和 可用性 则是二选一:假设注册中心 A 与注册中心 B、C 通信中断。某服务向 A 注册/注销,B、C 不知道;反之亦然。

  • 保留一致性 :为达到一致性,将节点数少的分区 A 停止对外服务,直到恢复网络,那么此时 A 是不可用的。
  • 保留可用性 :为达到可用,不同的网络分区同时对外提供服务,不同分区存在信息差,那么此时必是不一致的。

因此对于注册中心集群,AP、CP 架构二选一。

四、常见注册中心

  • Zookeeper:Apache 开源。遵循 CP 原则。节点集群有主从之分,当主节点故障,会重新选举主节点,这个过程有短暂时间不可用。

  • Eureka:Netflix 开源。虽然 2.0 版本已停止维护,但依然是 Spring Cloud 服务注册/发现的默认实现。遵循 AP 原则。每个节点均等

  • Nacos:Alibaba 开源。遵循 CP 或 AP,默认 AP。

    虽然 AP 可能会拿到没有即时更新的错误信息,但总比无法提供服务的 CP 强,比如高并发的场景。

五、搭建 Eureka 服务

1、搭建注册中心服务端

目的:提供服务注册、发现、健康检查能力。

(继【微服务】环境与工程搭建)

1、创建 Eureka 服务端子模块。

2、引入服务端依赖:

XML 复制代码
<dependency>
 <groupId>org.springframework.cloud</groupId>
 <artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
</dependency>

再加一个 maven 插件打包

3、yml 配置 Eureka:

java 复制代码
server:
  port: 10010
spring:
  application:
    name: eureka-server
eureka:
  instance:
    hostname: localhost
  client:
    fetch-registry: false # 表示是否从Eureka Server获取注册信息,默认为true.因为这是一个单点的Eureka Server,不需要同步其他的Eureka Server节点的数据,这里设置为false
    register-with-eureka: false # 表示是否将自己注册到Eureka Server,默认为true.由于当前应用就是Eureka Server,故而设置为false.
    service-url:
      # 设置Eureka Server的地址,查询服务和注册服务都需要依赖这个地址
      defaultZone: http://${eureka.instance.hostname}:${server.port}/eureka/

4、启动类开启 Eureka 功能:

java 复制代码
@EnableEurekaServer

5、启动项目,访问 Eureka 注册中心可视化管理页面:

XML 复制代码
http://${eureka.instance.hostname}:${server.port}

# 注意不是
http://${eureka.instance.hostname}:${server.port}/eureka/

2、服务注册

目的:把 product-service(服务提供者)注册到 eureka 服务端中。

1、引入 Eureka 客户端依赖:

XML 复制代码
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
        </dependency>

2、yml 配置 Eureka:

XML 复制代码
# 应用名称,后面远程调用时 通过应用名称 获取实例信息
spring:
  application:
    name: product-service

eureka:
  client:
    service-url:
      defaultZone: http://127.0.0.1:10010/eureka/

3、启动服务,查看管理页面:

3、服务发现

目的:order-service 从 eureka 服务端拉取 product-service 的服务信息,进行远程调用。

1、引入 Eureka 客户端依赖。

2、配置 Eureka。

3、修改远程调用方式:

java 复制代码
// 1. 注入服务发现
@Resource
private DiscoveryClient discoveryClient;

// 2. 从注册中心获取服务信息

//从Eureka中获取服务列表,指定要查询的服务名
List<ServiceInstance> instances = discoveryClient.getInstances("product-service");
// 因为目前只有一个商品服务,所以这里取第一个
String uri = instances.get(0).getUri().toString();
// 构造访问商品服务的 url
String url = uri+"/product/"+orderInfo.getProductId();

4、启动服务,查看管理页面:

查询订单服务,远程调用商品服务:

相关推荐
KYGALYX1 小时前
服务异步通信
开发语言·后端·微服务·ruby
yunteng5212 小时前
通用架构(同城双活)(单点接入)
架构·同城双活·单点接入
麦聪聊数据3 小时前
Web 原生架构如何重塑企业级数据库协作流?
数据库·sql·低代码·架构
程序员侠客行4 小时前
Mybatis连接池实现及池化模式
java·后端·架构·mybatis
陈桴浮海4 小时前
Kustomize实战:从0到1实现K8s多环境配置管理与资源部署
云原生·容器·kubernetes
bobuddy5 小时前
射频收发机架构简介
架构·射频工程
桌面运维家5 小时前
vDisk考试环境IO性能怎么优化?VOI架构实战指南
架构
一个骇客7 小时前
让你的数据成为“操作日志”和“模型饲料”:事件溯源、CQRS与DataFrame漫谈
架构
ShiLiu_mtx7 小时前
k8s - 7
云原生·容器·kubernetes
鹏北海-RemHusband8 小时前
从零到一:基于 micro-app 的企业级微前端模板完整实现指南
前端·微服务·架构