SpringCloud—02—初级之服务注册中心

提前预知

学习一种技术最好的方式是:视屏+官方文档!!!!!

官网文档地址:https://cloud.spring.io/spring-cloud-static/Hoxton.SR1/reference/htmlsingle/

注意:这个是H版的SR1对应的文档,如果想看其他版本的直接更改链接中的版本号即可,例如查看Hoxton.SR2版本,如下:

https://cloud.spring.io/spring-cloud-static/Hoxton.SR2/reference/htmlsingle/

代码地址:https://gitee.com/aismall/spring-cloud

安装虚拟机传送门:https://blog.csdn.net/weixin_45583303/article/details/110294058

安装zookeeper传送门:https://blog.csdn.net/weixin_45583303/article/details/119782963

本次笔记对应的课程为尚硅谷的Springcloud教程,课程地址:springcloud课程地址

课程包含:springcloud+springcloud alibaba

课程分为四个等级(不一定要一次性学完,你懂得!!!):

  • 零:1~4
  • 初:5~9
  • 中:10~16
  • 高:17~21

课程大纲:
在这里插入图片描述
一点一点的学习下面这些组件!!!!!
在这里插入图片描述

05、Eureka服务注册与发现

5.1、Eureka基础知识

在这里插入图片描述
上面的这种方式会有什么问题?

  • 如果是一对一的消费和提供,问题不大,就算一对多,只要消费端不太多,问题都不大。
  • 如果消费端的服务很多,也就是有很多消费服务要调用同一个微服务提供者就会出问题,这个微服务提供者有没有能力提供这么多的服务,这时候就需要进行服务治理

什么是服务治理?

  • Spring Cloud封装了Netflix 公司开发的Eureka模块来实现服务治理。
  • 在传统的RPC远程调用框架中,管理每个服务与服务之间依赖关系比较复杂,所以需要使用服务治理,管理服务于服务之间依赖关系。
  • 服务治理可以实现服务调用负载均衡容错服务发现服务注册等。

什么是服务注册与发现?

  • Eureka采用了CS的设计架构Eureka Sever作为服务注册功能的服务器,它是服务注册中心。
  • 而系统中的其他微服务,使用Eureka的客户端连接到 Eureka Server并维持心跳连接。
  • 这样系统的维护人员就可以通过Eureka Server来监控系统中各个微服务是否正常运行。
  • 在服务注册与发现中,有一个注册中心,当服务器启动的时候,会把当前自己服务器的信息比如服务地址通讯地址等以别名方式注册到注册中心上,另一方(消费者服务提供者)以该别名的方式去注册中心上获取到实际的服务通讯地址,然后再实现本地RPC调用。

RPC远程调用框架核心设计思想:

  • 在于注册中心,因为使用注册中心管理每个服务与服务之间的一个依赖关系(服务治理概念)。
  • 在任何RPC远程框架中,都会有一个注册中心存放服务地址相关信息(接口地址)
    在这里插入图片描述

Eureka包含两个组件:Eureka ServerEureka Client

Eureka Server提供服务注册服务:

  • 各个微服务节点通过配置启动后,会在EurekaServer中进行注册,这样EurekaServer中的服务注册表中将会存储所有可用服务节点的信息,服务节点的信息可以在界面中直观看到。

EurekaClient通过注册中心进行访问:

  • 它是一个Java客户端,用于简化Eureka Server的交互,客户端同时也具备一个内置的、使用轮询(round-robin)负载算法的负载均衡器。
  • 在应用启动后,将会向Eureka Server发送心跳(默认周期为30秒)。
  • 如果Eureka Server在多个心跳周期内没有接收到某个节点的心跳,EurekaServer将会从服务注册表中把这个服务节点(EurekaClient)移除(默认90秒)

5.2、EurekaServer服务端安装

IDEA生成eurekaServer端服务注册中心,类似物业公司:

也就是创建一个用于注册微服务服务,熟悉的5个步骤:

1、创建名为cloud-eureka-server7001的Maven工程

2.修改pom.xml

<!-- eureka以前的老版本(2018) -->
<dependency>
    <groupid>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-eureka</artifactId>
</dependency>

<!-- 现在新版本(2020.2),我们使用最新的-->
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
</dependency>
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <artifactId>SpringCloud02</artifactId>
        <groupId>com.aismall</groupId>
        <version>1.0-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>cloud-eureka-server7001</artifactId>

    <dependencies>
        <!--eureka-server-->
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
        </dependency>
        <!-- 引入自己定义的api通用包,可以使用Payment支付Entity -->
        <dependency>
            <groupId>com.aismall</groupId>
            <artifactId>cloud-api-commons</artifactId>
            <version>${project.version}</version>
        </dependency>
        <!--boot web actuator-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-actuator</artifactId>
        </dependency>
        <!--一般通用配置-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-devtools</artifactId>
            <scope>runtime</scope>
            <optional>true</optional>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
        </dependency>
    </dependencies>
</project>

3、写配置

server:
  port: 7001

eureka:
  instance:
    hostname: locathost #eureka服务端的实例名称
  client:
    #false表示不向注册中心注册自己。
    register-with-eureka: false
    #false表示自己端就是注册中心,我的职责就是维护服务实例,并不需要去检索服务
    fetch-registry: false
    service-url:
      #设置与Eureka server交互的地址查询服务和注册服务都需要依赖这个地址。
      defaultZone: http://${eureka.instance.hostname}:${server.port}/eureka/

4、主启动

@SpringBootApplication
@EnableEurekaServer
public class EurekaMain7001 {
    public static void main(String[] args) {
        SpringApplication.run(EurekaMain7001.class, args);
    }
}

5、测试运行EurekaMain7001

  • 浏览器输入:http://localhost:7001
  • 回车:会查看到Spring Eureka服务主页。
    在这里插入图片描述

5.3、单机Eureka构建步骤

5.3.1、支付微服务8001入驻进EurekaServer

EurekaClient端:将cloud-provider-payment8001注册进EurekaServer成为服务提供者provider,类似学校对外提供授课服务,修改cloud-provider-payment8001微服务。

1、改POM:添加spring-cloud-starter-netflix-eureka-client依赖

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>

2、改YML:添加eureka节点的配置

spring:
  application:
    name: cloud-payment-service				# 显示的应用名称
    
eureka:
  client:
    #表示是否将自己注册进Eurekaserver默认为true。
    register-with-eureka: true
    #是否从EurekaServer抓取已有的注册信息,默认为true。单节点无所谓,集群必须设置为true才能配合ribbon使用负载均衡
    fetchRegistry: true
    service-url:
      defaultZone: http://localhost:7001/eureka

3、改主启动:在启动类上添加一个注解

@EnableEurekaClient//<-----添加该注解

4、测试

  • 启动cloud-provider-payment8001cloud-eureka-server7001工程。

  • 浏览器输入:http://localhost:7001/
    在这里插入图片描述

  • 注意:主页内的Instances currently registered with Eureka会显示cloud-provider-payment8001的配置文件application.yml设置的应用名cloud-payment-service

5、自我保护机制

  • EMERGENCY! EUREKA MAY BE INCORRECTLY CLAIMING INSTANCES ARE UP WHEN THEY’RE NOT. RENEWALS ARELESSER THAN THRESHOLD AND HENCFT ARE NOT BEING EXPIRED JUST TO BE SAFE.

  • 紧急情况!EUREKA可能错误地声称实例在没有启动的情况下启动了,续订小于阈值,因此实例不会为了安全而过期。

5.3.2、订单微服务80入驻进EurekaServer

EurekaClient端:将cloud-consumer-order80注册进EurekaServer成为服务提供者provider,类似学校对外提供授课服务,修改cloud-consumer-order80微服务。

1、改POM:添加spring-cloud-starter-netflix-eureka-client依赖

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>

2、改YML:添加eureka节点的配置

spring:
  application:
    name: cloud-order-service
    
eureka:
  client:
    #表示是否将自己注册进Eurekaserver默认为true。
    register-with-eureka: true
    #是否从EurekaServer抓取已有的注册信息,默认为true。单节点无所谓,集群必须设置为true才能配合ribbon使用负载均衡
    fetchRegistry: true
    service-url:
      defaultZone: http://localhost:7001/eureka

3、改主启动:在启动类上添加一个注解

@EnableEurekaClient//<-----添加该注解

4、测试

  • 启动cloud-eureka-server7001cloud-provider-payment8001cloud-consumer-order80这三工程。
  • 浏览器输入:http://localhost:7001/
    在这里插入图片描述

5.4、集群Eureka构建步骤

5.4.1、Eureka集群原理说明

在这里插入图片描述

  • 服务注册:将服务信息注册进注册中心

  • 服务发现:从注册中心上获取服务信息

  • 实质:key=服务名value=服务地址

执行步骤:

  • 1、先启动eureka注主册中心

  • 2、启动服务提供者Payment支付服务

  • 3、支付服务启动后会把自身信息(比服务地址以别名方式注进eureka服务器

  • 4、消费者Order服务在需要调用接口时,使用服务别名去注册中心获取实际的RPC远程调用地址

  • 5、消费者调用地址后,底层实际上是利用HttpClient技术实现远程调用

  • 6、消费者获得服务地址后会缓存在本地jvm内存中,默认每间隔30秒更新—次服务调用地址

问题:微服务RPC远程服务调用最核心的是什么

  • 高可用:试想你的注册中心只有一个only one,万一它出故障了,会导致整个为服务环境不可用。
  • 解决办法:搭建Eureka注册中心集群,实现负载均衡+故障容错
    在这里插入图片描述
5.4.2、Eureka集群环境构建

创建cloud-eureka-server7002工程,参考 :5.2、EurekaServer服务端安装

找到C:\Windows\System32\drivers\etc路径下的hosts文件,修改映射配置添加进hosts文件

127.0.0.1 eureka7001.com
127.0.0.1 eureka7002.com

修改cloud-eureka-server7001配置文件

server:
  port: 7001
  
eureka:
  instance:
    hostname: eureka7001.com		 #eureka服务端的实例名称
  client:
    register-with-eureka: false     #false表示不向注册中心注册自己。
    fetch-registry: false     		#false表示自己端就是注册中心,我的职责就是维护服务实例,并不需要去检索服务
    service-url:
    #集群指向其它eureka
      defaultZone: http://eureka7002.com:7002/eureka/
    #单机就是7001自己
      #defaultZone: http://eureka7001.com:7001/eureka/

修改cloud-eureka-server7002配置文件

server:
  port: 7002

eureka:
  instance:
    hostname: eureka7002.com #eureka服务端的实例名称
  client:
    register-with-eureka: false     #false表示不向注册中心注册自己。
    fetch-registry: false     #false表示自己端就是注册中心,我的职责就是维护服务实例,并不需要去检索服务
    service-url:
    #集群指向其它eureka
      defaultZone: http://eureka7001.com:7001/eureka/
    #单机就是7002自己
      #defaultZone: http://eureka7002.com:7002/eureka/
5.4.3、订单支付两微服务注册进Eureka集群

支付服务8001微服务订单服务80微服务发布到上面2台Eureka集群配置中

将它们的配置文件的eureka.client.service-url.defaultZone进行修改

eureka:
  client:
    #表示是否将自己注册进Eurekaserver默认为true。
    register-with-eureka: true
    #是否从EurekaServer抓取已有的注册信息,默认为true。单节点无所谓,集群必须设置为true才能配合ribbon使用负载均衡
    fetchRegistry: true
    service-url:
      defaultZone: http://eureka7001.com:7001/eureka, http://eureka7002.com:7002/eureka

测试

先要启动EurekaServer7001/7002服务
再要启动服务提供者provider:8001
再要启动消费者:80
浏览器输入:http://localhost/consumer/payment/get/1
5.4.4、支付微服务集群配置

为了支付这个微服务的高可用,我们对支付这个微服务进行集群的搭建,新建cloud-provider-payment8002微服务,过程参考cloud-provicer-payment8001微服务搭建过程

1、新建cloud-provider-payment8002
2、改POM
3.写YML:端口号为:8002
4.主启动
5.业务类
6.修改8001/8002的Controller类,添加serverPort属性,为了方便知道那个支付(8001/8002)微服务提供的服务。

@RestController
@Slf4j
public class PaymentController{
    @Resource
    private PaymentService paymentService;
    @Value("${server.port}")
    private String serverPort;        //添加serverPort

    @PostMapping(value = "/payment/create")
    public CommonResult create(@RequestBody Payment payment) {
        int result = paymentService.create(payment);
        log.info("*****插入结果:"+result);

        if(result > 0) {
            return new CommonResult(200,"插入数据库成功,serverPort: "+serverPort, result);
        }else{
            return new CommonResult(444,"插入数据库失败",null);
        }
    }

    @GetMapping(value = "/payment/get/{id}")
    public CommonResult<Payment> getPaymentById(@PathVariable("id") Long id) {
        Payment payment = paymentService.getPaymentById(id);

        if(payment != null) {
            return new CommonResult(200,"查询成功,serverPort: "+serverPort,payment);
        }else{
            return new CommonResult(444,"没有对应记录,查询ID: "+id,null);
        }
    }
}
5.4.5、消费订单微服的负载均衡

cloud-consumer-order80订单服务访问地址不能写死,修改OrderController类

@Slf4j
@RestController
public class OrderController {

    //public static final String PAYMENT_URL = "http://localhost:8001";
    // 因为两个支付微服务注册进Eureka服务器的别名都是:cloud-pyment-service
    public static final String PAYMENT_URL = "http://CLOUD-PAYMENT-SERVICE";
    
    ...
}

使用@LoadBalanced注解赋予RestTemplate负载均衡的能力,修改ApplicationContextConfig类:

@Configuration
public class ApplicationContextConfig {

    @Bean
    @LoadBalanced//使用@LoadBalanced注解赋予RestTemplate负载均衡的能力
    public RestTemplate getRestTemplate(){
        return new RestTemplate();
    }

}

测试

先要启动EurekaServer7001/7002服务

再要启动服务提供者provider:8001/8002服务

浏览器输入:http://localhost/consumer/payment/get/1

结果:负载均衡效果达到,8001/8002端口交替出现

RibbonEureka整合后Consumer可以直接调用服务而不用再关心地址和端口号,且该服务还有负载功能。
访问Eureka节点1客户端:http://localhost:7001/
访问Eureka节点2客户端:http://localhost:7002/

5.5、actuator微服务信息完善

主机名称:服务名称修改(也就是将IP地址,换成可读性高的名字)

  • 修改cloud-provider-payment8001的配置文件:YML

    eureka:
      ...
      instance:
        instance-id: payment8001 #添加此处
    
  • 修改cloud-provider-payment8002的配置文件:YML

    eureka:
      ...
      instance:
        instance-id: payment8002 #添加此处
    
  • 修改之后:eureka主页将显示payment8001,payment8002代替原来显示的IP地址。

访问信息有IP信息提示,(就是将鼠标指针移至payment8001,payment8002名下,会有IP地址提示,提示信息在页面的左下角

  • 修改cloud-provider-payment8001的配置文件:YML

    eureka:
      ...
      instance:
        instance-id: payment8001 
        prefer-ip-address: true #添加此处
    
  • 修改cloud-provider-payment8002的配置文件:YML

    eureka:
      ...
      instance:
        instance-id: payment8002 
        prefer-ip-address: true #添加此处
    
访问Eureka节点1客户端:http://localhost:7001/
访问Eureka节点2客户端:http://localhost:7002/

5.6、服务发现Discovery

对于注册进eureka里面的微服务,可以通过服务发现来获得该服务的信息

修改cloud-provider-payment8001的Controller类:

@RestController
@Slf4j
public class PaymentController{
	...
    
    @Resource
    private DiscoveryClient discoveryClient;

    ...

    @GetMapping(value = "/payment/discovery")
    public Object discovery()
    {
        List<String> services = discoveryClient.getServices();
        for (String element : services) {
            log.info("*****element: "+element);
        }

        List<ServiceInstance> instances = discoveryClient.getInstances("CLOUD-PAYMENT-SERVICE");
        for (ServiceInstance instance : instances) {
            log.info(instance.getServiceId()+"\t"+instance.getHost()+"\t"+instance.getPort()+"\t"+instance.getUri());
        }

        return this.discoveryClient;
    }
}

修改cloud-provider-payment8001的启动类:

@SpringBootApplication
@EnableEurekaClient
@EnableDiscoveryClient//添加该注解
public class PaymentMain001 {
    public static void main(String[] args) {
        SpringApplication.run(PaymentMain001.class, args);
    }
}

自测

- 先要启动EurekaSeryer

- 再启动8001主启动类,需要稍等一会儿

- 浏览器输入:http://localhost:8001/payment/discovery

- 浏览器输出:{"services":["cloud-payment-service"],"order":0}

- 后台输出:	
	*****element: cloud-payment-service
	CLOUD-PAYMENT-SERVICE	192.168.199.218	8001	http://192.168.199.218:8001

5.7、Eureka自我保护理论知识

5.7.1、概述

保护模式主要用于一组客户端和Eureka Server之间存在网络分区场景下的保护。一旦进入保护模式,Eureka Server将会尝试保护其服务注册表中的信息,不再删除服务注册表中的数据,也就是不会注销任何微服务。

如果在Eureka Server的首页看到以下这段提示,则说明Eureka进入了保护模式:

  • EMERGENCY! EUREKA MAY BE INCORRECTLY CLAIMING INSTANCES ARE UP WHEN THEY’RE NOT. RENEWALS ARE LESSER THANTHRESHOLD AND HENCE THE INSTANCES ARE NOT BEING EXPIRED JUSTTO BE SAFE.

导致原因?

  • 一句话:某时刻某一个微服务不可用了,Eureka不会立刻清理,依旧会对该微服务的信息进行保存。
  • 属于CAP里面的AP分支。

为什么会产生Eureka自我保护机制?

  • 为了EurekaClient可以正常运行,防止与EurekaServer网络不通情况下,EurekaServer不会立刻将EurekaClient服务剔除

什么是自我保护模式?

  • 默认情况下,如果EurekaServer在一定时间内没有接收到某个微服务实例的心跳,EurekaServer将会注销该实例(默认90秒)。
  • 但是当网络分区故障发生(延时、卡顿、拥挤)时,微服务与EurekaServer之间无法正常通信,以上行为可能变得非常危险了——因为微服务本身其实是健康的,此时本不应该注销这个微服务。
  • Eureka通过自我保护模式来解决这个问题——当EurekaServer节点在短时间内( 90秒中)丢失过多客户端时(可能发生了网络分区故障),那么这个节点就会进入自我保护模式

自我保护机制:默认情况下EurekaClient定时向EurekaServer端发送心跳包

  • 如果Eureka在server端在一定时间内(默认90秒)没有收到EurekaClient发送心跳包,便会直接从服务注册列表中剔除该服务,但是在短时间内丢失了大量的服务实例心跳,这时候Eurekaserver会开启自我保护机制,不会剔除该服务(该现象可能出现在如果网络不通但是EurekaClient为出现宕机,此时如果换做别的注册中心如果一定时间内没有收到心跳会将剔除该服务,这样就出现了严重失误,因为客户端还能正常发送心跳,只是网络延迟问题,而保护机制是为了解决此问题而产生的)。

  • 在自我保护模式中,Eureka Server会保护服务注册表中的信息,不再注销任何服务实例。

  • 它的设计哲学就是宁可保留错误的服务注册信息,也不盲目注销任何可能健康的服务实例。一句话讲解:好死不如赖活着。

  • 综上,自我保护模式是一种应对网络异常的安全保护措施。它的架构哲学是宁可同时保留所有微服务(健康的微服务和不健康的微服务都会保留)也不盲目注销任何健康的微服务。使用自我保护模式,可以让Eureka集群更加的健壮、稳定。

5.7.2、怎么禁止自我保护

例如:在eurekaServer服务端7001处设置关闭自我保护机制

注意:出厂默认,自我保护机制是开启的

使用eureka.server.enable-self-preservation = false可以禁用自我保护模式

eureka:
  ...
  server:
    #关闭自我保护机制,保证不可用服务被及时踢除
    enable-self-preservation: false
    eviction-interval-timer-in-ms: 2000

关闭效果:spring-eureka主页会显示出一句

  • THE SELF PRESERVATION MODE IS TURNED OFF. THIS MAY NOT PROTECT INSTANCE EXPIRY IN CASE OF NETWORK/OTHER PROBLEMS.

设置一下:生产者客户端eureakeClient端8001

  • 默认:

    eureka.instance.lease-renewal-interval-in-seconds=30
    
    eureka.instance.lease-expiration-duration-in-seconds=90
    
    eureka:
      ...
      instance:
        instance-id: payment8001
        prefer-ip-address: true
        #心跳检测与续约时间
        #开发时设置小些,保证服务关闭后注册中心能即使剔除服务
        #Eureka客户端向服务端发送心跳的时间间隔,单位为秒(默认是30秒)
        lease-renewal-interval-in-seconds: 1
        #Eureka服务端在收到最后一次心跳后等待时间上限,单位为秒(默认是90秒),超时将剔除服务
        lease-expiration-duration-in-seconds: 2
    

测试

- 70018001都配置完成
- 先启动7001再启动8001
- 结果:先关闭8001,马上被删除了

5.8、Eureka停更说明

链接:https://github.com/Netflix/eureka/wiki

Eureka 2.0 (Discontinued)

The existing open source work on eureka 2.0 is discontinued. The code base and artifacts that were released as part of the existing repository of work on the 2.x branch is considered use at your own risk.

Eureka 1.x is a core part of Netflix’s service discovery system and is still an active project.

由于Eureka已经停止更新了,我们后面使用ZooKeeper代替Eureka功能

06、Zookeeper服务注册与发现

6.1、Zookeeper概述

zookeeper安装参考链接:https://blog.csdn.net/weixin_45583303/article/details/119782963

zookeeper是什么?
官方文档上这么解释zookeeper,它是一个分布式服务框架,是Apache Hadoop 的一个子项目,它主要是用来解决分布式应用中经常遇到的一些数据管理问题,如:统一命名服务、状态同步服务、集群管理、分布式应用配置项的管理等。

上面的解释有点抽象,简单来说zookeeper=文件系统+监听通知机制

文件系统Zookeeper维护一个类似文件系统的数据结构
在这里插入图片描述

每个子目录项如 NameService 都被称作为 znode(目录节点),和文件系统一样,我们能够自由的增加、删除znode,在一个znode下增加、删除子znode,唯一的不同在于znode是可以存储数据的。

  • 有四种类型的znode:
PERSISTENT:持久化目录节点
- 客户端与zookeeper断开连接后,该节点依旧存在

PERSISTENT_SEQUENTIAL:持久化顺序编号目录节点
- 客户端与zookeeper断开连接后,该节点依旧存在,只是Zookeeper给该节点名称进行顺序编号

EPHEMERAL:临时目录节点
- 客户端与zookeeper断开连接后,该节点被删除

EPHEMERAL_SEQUENTIAL:临时顺序编号目录节点
- 客户端与zookeeper断开连接后,该节点被删除,只是Zookeeper给该节点名称进行顺序编号

监听通知机制:客户端注册监听它关心的目录节点,当目录节点发生变化(数据改变、被删除、子目录节点增加删除)时,zookeeper会通知客户端,就这么简单,下面我们看看Zookeeper能做点什么呢?

Zookeeper能做什么?

  • zookeeper功能非常强大,可以实现诸如分布式应用配置管理、统一命名服务、状态同步服务、集群管理等功能,我们这里拿比较简单的分布式应用配置管理为例来说明。

  • 假设我们的程序是分布式部署在多台机器上,如果我们要改变程序的配置文件,需要逐台机器去修改,非常麻烦,现在把这些配置全部放到zookeeper上去,保存在 zookeeper 的某个目录节点中,然后所有相关应用程序对这个目录节点进行监听,一旦配置信息发生变化,每个应用程序就会收到 zookeeper 的通知,然后从 zookeeper 获取新的配置信息应用到系统中。
    在这里插入图片描述

6.2、支付服务注册进zookeeper

注册中心使用:Zookeeper

zookeeper是一个分布式协调工具,可以实现注册中心功能

注意:

  • 如果连接不上zookeeper服务器,可以查看zookeeper服务的端口号时候被暴露出来,也可以直接关闭Linux系统的防火墙
  • zookeeper默认的端口为2181,可以使用命令查看此端口对应的zookeeper服务是否启动:ps -ef|grep 2181
  • 启动zookeeper命令:./zkServer.sh start
  • 我们linux虚拟机的ip为:192.168.1.101

服务提供者
1、建名为cloud-provider-payment8004的Maven工程。

2.POM

<dependencies>
    <!-- SpringBoot整合Web组件 -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    <dependency><!-- 引入自己定义的api通用包,可以使用Payment支付Entity -->
        <groupId>com.aismall</groupId>
        <artifactId>cloud-api-commons</artifactId>
        <version>${project.version}</version>
    </dependency>
    <!-- SpringBoot整合zookeeper客户端 -->
    <dependency>
        <groupId>org.springframework.cloud</groupId>
        <artifactId>spring-cloud-starter-zookeeper-discovery</artifactId>
        <!--先排除自带的zookeeper3.5.3 防止与3.5.8起冲突-->
        <exclusions>
            <exclusion>
                <groupId>org.apache.zookeeper</groupId>
                <artifactId>zookeeper</artifactId>
            </exclusion>
        </exclusions>
    </dependency>
    <!--添加zookeeper3.5.8版本-->
    <dependency>
        <groupId>org.apache.zookeeper</groupId>
        <artifactId>zookeeper</artifactId>
        <version>3.5.8</version>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-devtools</artifactId>
        <scope>runtime</scope>
        <optional>true</optional>
    </dependency>
    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <optional>true</optional>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-test</artifactId>
        <scope>test</scope>
    </dependency>
</dependencies>

3、YML

#8004表示注册到zookeeper服务器的支付服务提供者端口号
server:
  port: 8004
  
#服务别名----注册zookeeper到注册中心名称
spring:
  application:
    name: cloud-provider-payment
  cloud:
    zookeeper:
      connect-string: 192.168.1.101:2181 

4、主启动

@SpringBootApplication
@EnableDiscoveryClient//该注解用于向使用consul或者zookeeper作为注册中心时注册服务
public class PaymentMain8004 {
    public static void main(String[] args) {
        SpringApplication.run(PaymentMain8004.class, args);
    }
}

5、业务类

  • Controller类
  • 注意:此处我们没有读写数据库,返回的知识一个字符串,就不用写那么多的类了。
package com.aismall.springcloud.controller;

@RestController
@Slf4j
public class PaymentController
{
    @Value("${server.port}")
    private String serverPort;

    @RequestMapping(value = "/payment/zk")
    public String paymentzk()
    {
        return "springcloud with zookeeper: "+serverPort+"\t"+ UUID.randomUUID().toString();
    }
}

6、启动8004注册进zookeeper(要先启动zookeeper的server)

  • 验证测试:浏览器 - http://localhost:8004/payment/zk

  • 验证测试2 :接着用zookeeper客户端操作

先介绍一下zookeeper的客户端:
- ZooKeeper命令行工具类似于Linux的shell环境,不过功能肯定不及shell啦,但是使用它我们可以简单的
-ZooKeeper进行访问,数据创建,数据修改等操作.  
- 使用 zkCli.sh连接到 ZooKeeper服务,连接成功后,系统会输出 ZooKeeper 的相关环境以及配置信息。

命令行工具的一些简单操作如下:
- 1. 显示根目录下文件: ls /  使用 ls 命令来查看当前 ZooKeeper 中所包含的内容
- 2. 显示根目录下文件: ls2 / 查看当前节点数据并能看到更新次数等数据
- 3. 创建文件,并设置初始内容:create /zk "test" 创建一个新的 znode节点zk以及与它关联的字符串
- 4. 获取文件内容: get /zk  确认znode 是否包含我们所创建的字符串
- 5. 修改文件内容: set /zk "zkbak" 对 zk 所关联的字符串进行设置
- 6. 删除文件: delete /zk 将刚才创建的 znode 删除
- 7. 退出客户端: quit
- 8. 帮助命令: help
[root@localhost apache-zookeeper-3.5.8-bin]# ./bin/zkCli.sh
......
[zk: localhost:2181(CONNECTED) 1] ls /
[services, zookeeper]
[zk: localhost:2181(CONNECTED) 2]  ls /services/cloud-provider-payment
[ecd49551-0e16-405e-99a9-0ced4e23f4e7]
[zk: localhost:2181(CONNECTED) 3] get /services/cloud-provider-payment/ecd49551-0e16-405e-99a9-0ced4e23f4e7
{"name":"cloud-provider-payment","id":"ecd49551-0e16-405e-99a9-0ced4e23f4e7","address":"192.168.1.1","port":8004,"sslPort":null,"payload":{"@class":"org.springframework.cloud.zookeeper.discovery.ZookeeperInstance","id":"application-1","name":"cloud-provider-payment","metadata":{}},"registrationTimeUTC":1629346059566,"serviceType":"DYNAMIC","uriSpec":{"parts":[{"value":"scheme","variable":true},{"value":"://","variable":false},{"value":"address","variable":true},{"value":":","variable":false},{"value":"port","variable":true}]}}
  • 使用JSON格式化工具格式化get命令得到的内容:
{
    "name":"cloud-provider-payment",
    "id":"ecd49551-0e16-405e-99a9-0ced4e23f4e7",
    "address":"192.168.1.1",
    "port":8004,
    "sslPort":null,
    "payload":{
        "@class":"org.springframework.cloud.zookeeper.discovery.ZookeeperInstance",
        "id":"application-1",
        "name":"cloud-provider-payment",
        "metadata":{

        }
    },
    "registrationTimeUTC":1629346059566,
    "serviceType":"DYNAMIC",
    "uriSpec":{
        "parts":[
            {
                "value":"scheme",
                "variable":true
            },
            {
                "value":"://",
                "variable":false
            },
            {
                "value":"address",
                "variable":true
            },
            {
                "value":":",
                "variable":false
            },
            {
                "value":"port",
                "variable":true
            }
        ]
    }
}

6.3、临时还是持久节点

ZooKeeper的服务节点是临时节点,没有Eureka那含情脉脉。

EPHEMERAL:临时目录节点
- 客户端与zookeeper断开连接后,该节点被删除

6.4、订单服务注册进zookeeper

1、新建cloud-consumerzk-order80

2.POM

<dependencies>
    <!-- SpringBoot整合Web组件 -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    <!-- SpringBoot整合zookeeper客户端 -->
    <dependency>
        <groupId>org.springframework.cloud</groupId>
        <artifactId>spring-cloud-starter-zookeeper-discovery</artifactId>
        <!--先排除自带的zookeeper-->
        <exclusions>
            <exclusion>
                <groupId>org.apache.zookeeper</groupId>
                <artifactId>zookeeper</artifactId>
            </exclusion>
        </exclusions>
    </dependency>
    <!--添加zookeeper3.5.8版本-->
    <dependency>
        <groupId>org.apache.zookeeper</groupId>
        <artifactId>zookeeper</artifactId>
        <version>3.5.8</version>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-devtools</artifactId>
        <scope>runtime</scope>
        <optional>true</optional>
    </dependency>
    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <optional>true</optional>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-test</artifactId>
        <scope>test</scope>
    </dependency>
</dependencies>

3、YML

server:
  port: 80

#服务别名----注册zookeeper到注册中心名称
spring:
  application:
    name: cloud-consumer-order
  cloud:
    zookeeper:
      connect-string: 192.168.1.101:2181

4、主启动:(注意:两个80端口的服务不能同时启动

@SpringBootApplication
@EnableDiscoveryClient
public class OrderZKMain80 {
    public static void main(String[] args) {
        SpringApplication.run(OrderZKMain80.class, args);
    }
}

5、业务类

  • config类

    @Configuration
    public class ApplicationContextConfig
    {
        @Bean
        @LoadBalanced
        public RestTemplate getRestTemplate()
        {
            return new RestTemplate();
        }
    }
    
  • controller类

    @RestController
    @Slf4j
    public class OrderZKController
    {
        public static final String INVOKE_URL = "http://cloud-provider-payment";
    
        @Resource
        private RestTemplate restTemplate;
    
        @GetMapping(value = "/consumer/payment/zk")
        public String paymentInfo()
        {
            String result = restTemplate.getForObject(INVOKE_URL+"/payment/zk",String.class);
            return result;
        }
    }
    

6、测试
运行ZooKeeper服务端,cloud-consumerzk-order80cloud-provider-payment8004

打开ZooKeeper客户端:

[zk: localhost:2181(CONNECTED) 0] ls /
[services, zookeeper]
[zk: localhost:2181(CONNECTED) 1] ls /services
[cloud-consumer-order, cloud-provider-payment]

7、访问测试地址:http://localhost/consumer/payment/zk

07、Consul服务注册与发现

7.1、Consul简介

官网地址:https://www.consul.io/
下载地址:https://www.consul.io/downloads
官方文档:https://www.consul.io/docs/intro

在官方文档中的Intro to Consul这个章节,详细介绍了Consul,下面简单的介绍一下Consul:

  • Consul是一套开源的分布式服务发现和配置管理系统,由HashiCorp 公司用Go语言开发

  • 提供了微服务系统中的服务治理配置中心控制总线等功能,这些功能中的每一个都可以根据需要单独使用,也可以一起使用以构建全方位的服务网格,总之Consul提供了一种完整的服务网格解决方案。

它具有很多优点,包括:

  • 基于raft协议,比较简洁;
  • 支持健康检查;
  • 同时支持HTTP和DNS协议支持跨数据中心的WAN集群;
  • 提供图形界面;
  • 跨平台,支持Linux、Mac、Windows;

能干嘛?

  • 服务发现 -提供HTTP和DNS两种发现方式。
  • 健康监测 - 支持多种方式,HTTP、TCP、Docker、Shell脚本定制化
  • KV存储 - Key、Value的存储方式
  • 多数据中心 - Consul支持多数据中心
  • 可视化Web界面

如何使用,参考文档链接:https://www.springcloud.cc/spring-cloud-consul.html

7.2、安装并运行Consul

Consul支持很多版本,我们安装windows版本的,其实很简单,我们从官网先下载一个Consul,然后解压(里面就一个consul.exe),配置环境变量即可:
在这里插入图片描述
上面做完之后,打开终端,输入:consul
在这里插入图片描述
启动Cousul:consul agent -dev

web前端页面:http://localhost:8500/

7.3、服务提供者注册进Consul

1、新建Module支付服务:cloud-provider-consulpayment8006

2、改POM

<dependencies>
    <!-- 引入自己定义的api通用包,可以使用Payment支付Entity -->
    <dependency>
        <groupId>com.aismall</groupId>
        <artifactId>cloud-api-commons</artifactId>
        <version>${project.version}</version>
    </dependency>
    <!--SpringCloud consul-server -->
    <dependency>
        <groupId>org.springframework.cloud</groupId>
        <artifactId>spring-cloud-starter-consul-discovery</artifactId>
    </dependency>
    <!-- SpringBoot整合Web组件 -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-actuator</artifactId>
    </dependency>
    <!--日常通用jar包配置-->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-devtools</artifactId>
        <scope>runtime</scope>
        <optional>true</optional>
    </dependency>
    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <optional>true</optional>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-test</artifactId>
        <scope>test</scope>
    </dependency>
    <dependency>
        <groupId>cn.hutool</groupId>
        <artifactId>hutool-all</artifactId>
        <version>RELEASE</version>
        <scope>test</scope>
    </dependency>
    <dependency>
        <groupId>cn.hutool</groupId>
        <artifactId>hutool-all</artifactId>
        <version>RELEASE</version>
        <scope>test</scope>
    </dependency>
</dependencies>

3、YML

#consul服务端口号
server:
  port: 8006

spring:
  application:
    name: consul-provider-payment
# consul注册中心地址
  cloud:
    consul:
      host: localhost
      port: 8500
      discovery:
        #hostname: 127.0.0.1
        service-name: ${spring.application.name}

4、主启动类

@SpringBootApplication
@EnableDiscoveryClient
public class PaymentMain8006
{
    public static void main(String[] args) {
            SpringApplication.run(PaymentMain8006.class, args);
    }
}

5、业务类Controller

package com.aismall.springcloud.controller;
@RestController
@Slf4j
public class PaymentController
{
    @Value("${server.port}")
    private String serverPort;

    @RequestMapping(value = "/payment/consul")
    public String paymentConsul()
    {
        return "springcloud with consul: "+serverPort+"\t   "+ UUID.randomUUID().toString();
    }
}

6、验证测试,启动cloud-providerconsul-payment8006微服务:

浏览器:http://localhost:8006/payment/consul
浏览器:http://localhost:8500 - 会显示provider8006

在这里插入图片描述

7.4、服务消费者注册进Consul

1、新建Module消费服务:cloud-consumer-consulorder80

2.POM

<dependencies>
    <!--SpringCloud consul-server -->
    <dependency>
        <groupId>org.springframework.cloud</groupId>
        <artifactId>spring-cloud-starter-consul-discovery</artifactId>
    </dependency>
    <!-- SpringBoot整合Web组件 -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-actuator</artifactId>
    </dependency>
    <!--日常通用jar包配置-->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-devtools</artifactId>
        <scope>runtime</scope>
        <optional>true</optional>
    </dependency>
    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <optional>true</optional>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-test</artifactId>
        <scope>test</scope>
    </dependency>
</dependencies>

3、YML

###consul服务端口号
server:
  port: 80

spring:
  application:
    name: cloud-consumer-order
####consul注册中心地址
  cloud:
    consul:
      host: localhost
      port: 8500
      discovery:
        #hostname: 127.0.0.1
        service-name: ${spring.application.name}

4、主启动

@SpringBootApplication
@EnableDiscoveryClient //该注解用于向使用consul或者zookeeper作为注册中心时注册服务
public class OrderConsulMain80
{
    public static void main(String[] args) {
            SpringApplication.run(OrderConsulMain80.class, args);
    }
}

5、业务类

  • config类

    @Configuration
    public class ApplicationContextConfig
    {
        @Bean
        @LoadBalanced
        public RestTemplate getRestTemplate()
        {
            return new RestTemplate();
        }
    }
    
  • controller类

    @RestController
    @Slf4j
    public class OrderConsulController
    {
        public static final String INVOKE_URL = "http://consul-provider-payment";
    
        @Resource
        private RestTemplate restTemplate;
    
        @GetMapping(value = "/consumer/payment/consul")
        public String paymentInfo()
        {
            String result = restTemplate.getForObject(INVOKE_URL+"/payment/consul",String.class);
            return result;
        }
    }
    

6、测试

  • 运行consulcloud-providerconsul-payment8006cloud-consumerconsul-order80

  • 浏览器:http://localhost:8500/
    在这里插入图片描述

  • 浏览器: http://localhost/consumer/payment/consul

7.5、三个注册中心异同点

组件名语言CAP服务健康检查对外暴露接口SpringCloud集成
EurekaJavaAP可配支持HTTP
ConsulGoCP支持支持HTTP/DNS
ZookeeperJavaCP支持客户端已集成

CAP原理:

  • C:Consistency (强一致性)

  • A:Availability (可用性)

  • P:Partition tolerance (分区容错性)

  • 最多只能同时较好的满足两个。

CAP理论的核心是:一个分布式系统不可能同时很好的满足一致性可用性分区容错性这三个需求。

因此,根据CAP原理将NoSQL数据库分成了满足CA原则、满足CP原则和满足AP原则三大类:

  • CA - 单点集群,满足—致性,可用性的系统,通常在可扩展性上不太强大。
  • CP - 满足一致性,分区容忍必的系统,通常性能不是特别高。
  • AP - 满足可用性,分区容忍性的系统,通常可能对一致性要求低一些。

AP架构(Eureka

  • 当网络分区出现后,为了保证可用性,系统B可以返回旧值,保证系统的可用性。

  • 结论:违背了一致性C的要求,只满足可用性和分区容错,即AP
    在这里插入图片描述

CP架构(ZooKeeper/Consul

  • 当网络分区出现后,为了保证一致性,就必须拒接请求,否则无法保证一致性。

  • 结论:违背了可用性A的要求,只满足一致性和分区容错,即CP。
    在这里插入图片描述

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

彤彤的小跟班

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值