当前位置: 首页 > news >正文

微服务上下线动态感知实现的技术解析

序言

随着微服务架构的广泛应用,服务的动态管理和监控变得尤为重要。在微服务架构中,服务的上下线是一个常见的操作,如何实时感知这些变化,确保系统的稳定性和可靠性,成为了一个关键技术挑战。本文将深入探讨微服务上下线动态感知的实现方式,从技术基础、场景案例、解决思路和底层原理等多个维度进行阐述,并分别使用Java和Python进行演示介绍。

场景案例描述

场景一:业务系统微服务上下线挑战

某业务系统采用Spring Boot和Spring Cloud框架,服务发布流程中经常遇到以下问题:

  1. 过早销毁对象:服务正在处理请求时,由于服务实例被过早销毁,导致请求报错。
  2. 服务未及时下线:调用方未能及时感知服务已在下线,仍发送请求至已下线的服务实例,导致请求失败。
  3. 过早注册服务:服务未初始化完成即被注册到注册中心,导致接口响应时间突增甚至超时。

场景二:大规模微服务无损上下线需求

在云原生环境下,某客户在生产环境中使用Spring Cloud应用时,发现发布过程中出现了大量错误,如ServiceUnavailable。分析后发现,问题的根源在于某些消费者未能及时收到提供者的下线通知,导致请求仍然被发送到已下线的服务实例。

解决业务场景思路

针对上述场景,我们需要实现微服务的优雅上下线,确保服务在上下线过程中不会中断现有请求,并能及时通知调用方更新服务列表。以下是解决这些问题的关键思路:

优雅上线

  1. 资源初始化:通过预热功能实现服务资源的初始化。预热模块可以是可插拔的,业务方可以根据需要自定义预热逻辑,或者使用线上请求回放预热。
  2. 服务注册:在服务初始化完成后,再将其注册到注册中心,避免过早注册导致的接口响应时间问题。

优雅下线

  1. 取消注册与标记下线:服务实例在下线前,先向注册中心取消注册,并标记自己为下线状态。
  2. 请求处理:在标记为下线状态后,服务实例会阻塞一段时间(如5秒),以确保正在处理的请求能够成功返回。
  3. 负载剔除:服务调用方在收到下线标记后,将该实例从可用服务列表中剔除,确保后续请求不再打到该实例上。

实现无损上下线

  1. 主动注销与通知:在服务端内置HttpServer,提供/offline接口,用于接收主动注销的通知。在K8s的Prestop钩子中触发该接口,实现服务提前注销。
  2. 长连接维护:对于长连接框架(如Dubbo),可以在服务提供者中维护与服务消费者的连接集合,在收到offline命令后,向所有连接发送只读信号,确保不再接收新请求。
  3. 自适应等待:服务端在收到下线通知后,采用自适应等待策略,确保所有在途请求处理完成后再进行停机。

底层原理介绍

服务注册与发现

服务注册与发现是微服务架构中实现服务动态感知的基础。常见的服务注册中心有Eureka、Consul、Nacos等。

服务注册

当一个新的微服务实例启动时,它会向服务注册中心注册自己的信息,包括服务名称、IP地址、端口号等。这些信息将被存储在注册中心的目录中,供其他服务查询。

服务发现

其他服务实例通过查询注册中心,可以获取到目标服务的地址信息,从而实现服务的调用。注册中心通常提供RESTful API或客户端库,方便服务实例进行查询。

心跳机制

已注册的服务实例会定期向注册中心发送心跳包,以表明自己仍然存活。如果注册中心在一定时间内未收到某个服务实例的心跳包,将认为该实例已经下线,并将其从注册列表中移除。

事件通知机制

当服务注册中心感知到服务的上下线变化时,会通过事件通知机制及时通知订阅了该服务的其他微服务实例。这样,订阅者可以实时更新自己的本地缓存或服务列表,确保服务调用的准确性。

优雅上下线实现细节

优雅上线
  1. 预热逻辑
    • 自定义预热:业务方根据实际需求扩展预热逻辑,如加载缓存数据、初始化数据库连接等。
    • 线上请求回放预热:通过配置预热接口,拉取线上请求对本地服务进行预热。当接口调用达到预热次数后,服务再注册到注册中心。
  1. 服务注册组件
    • 在Spring Cloud Netflix基础上,定制开发GracefulServiceRegistration组件,负责触发预热逻辑和服务注册。
    • WarmUp接口用于定义预热逻辑,业务方可以实现该接口并注册到Spring容器中。
优雅下线
  1. 取消注册与标记下线
    • 在服务下线前,通过GracefulServiceRegistration组件取消注册,并标记服务为下线状态。
    • 使用InvokePlugin插件检查服务状态,如果服务处于下线中,则直接返回下线标记。
  1. 请求处理与等待
    • 服务实例在标记为下线状态后,会阻塞当前线程一段时间(如5秒),确保正在处理的请求能够成功返回。
    • 如果在此期间收到新请求,将直接返回下线标记。
  1. 负载剔除
    • 服务调用方在收到下线标记后,将该实例从可用服务列表中剔除。
    • 使用独立的任务线程处理失活队列,并维护可用注册表,确保后续请求不再打到已下线的实例上。

无损上下线技术实现

主动注销与通知
  1. 内置HttpServer
    • 在服务提供者进程中内置HttpServer,提供/offline接口用于接收主动注销的通知。
    • 在K8s的Prestop钩子中配置curl http://localhost:20001/offline触发主动注销。
  1. 长连接维护
    • 对于长连接框架(如Dubbo),在服务提供者中维护与服务消费者的连接集合。
    • 在收到offline命令后,向所有连接发送只读信号,确保不再接收新请求。
  1. 自适应等待
    • 服务端在收到下线通知后,采用自适应等待策略。
    • 统计并计算进入服务提供者和调用完成的流量,确保所有在途请求处理完成后再进行停机。

关键技术点解析

Zookeeper在动态感知中的应用

Zookeeper作为一种分布式协调服务,也可以用于实现微服务的动态感知。通过Zookeeper的watch机制,可以实时监控服务节点的变化。

  1. 节点创建与删除
    • 当服务上线时,在Zookeeper中创建一个临时节点表示该服务实例。
    • 当服务下线时,该临时节点会自动删除。
  1. 客户端监听
    • 客户端通过监听Zookeeper中的特定节点(如服务列表节点),可以实时感知服务实例的上下线。
    • 当监听到节点变化时,客户端会更新本地服务列表,确保服务调用的准确性。
Spring Cloud与Eureka的集成

Spring Cloud提供了对Eureka的集成支持,使得在Spring Boot应用中实现服务注册与发现变得非常简单。

  1. 依赖引入
    • 在Spring Boot项目的pom.xml文件中引入Spring Cloud Starter Netflix Eureka Client依赖。
  1. 配置Eureka客户端
    • application.ymlapplication.properties文件中配置Eureka客户端的相关属性,如服务名称、Eureka服务器地址等。
  1. 启用Eureka客户端
    • 在Spring Boot应用的启动类上添加@EnableEurekaClient注解,启用Eureka客户端功能。
  1. 服务注册与发现
    • 启动应用后,服务实例会自动向Eureka服务器注册。
    • 其他服务实例可以通过Eureka服务器查询目标服务的地址信息,实现服务调用。
K8s Prestop钩子的应用

在Kubernetes中,可以使用Prestop钩子在容器停止之前执行一些清理或通知操作。这对于实现微服务的无损下线非常有用。

  1. 配置Prestop钩子
    • 在Kubernetes的Pod定义文件中,为容器配置Prestop钩子。
    • 钩子可以是一个执行脚本或命令,用于在服务实例下线前进行必要的操作。
  1. 触发主动注销
    • 在Prestop钩子中触发服务实例的主动注销操作。
    • 通过调用服务提供者内置的/offline接口或向注册中心发送取消注册请求来实现。
  1. 等待请求处理完成
    • 在触发注销操作后,服务实例会进入等待状态,确保所有在途请求处理完成后再进行停机。

演示介绍

Java 演示

以下是一个使用Java和Spring Cloud Eureka实现微服务上下线动态感知的示例。

引入依赖
xml复制代码
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
配置Eureka客户端
yaml复制代码
spring:
application:
name: demo-service
eureka:
client:
service-url:
defaultZone: http://localhost:8761/eureka/
instance:
prefer-ip-address: true
lease-renewal-interval-in-seconds: 5 # 心跳间隔时间
lease-expiration-duration-in-seconds: 15 # 服务失效时间
启动类
java复制代码
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
@SpringBootApplication
@EnableEurekaClient
public class DemoServiceApplication {
public static void main(String[] args) {SpringApplication.run(DemoServiceApplication.class, args);}
}
控制器
java复制代码
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class DemoController {
@GetMapping("/hello")
public String hello() {
return "Hello from Demo Service!";}
}
启动Eureka服务器

如果没有现成的Eureka服务器,可以创建一个Spring Boot项目并配置Eureka服务器。

yaml复制代码
server:
port: 8761
eureka:
client:
register-with-eureka: false
fetch-registry: false
spring:
application:
name: eureka-server

在启动类上添加@EnableEurekaServer注解:

java复制代码
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.server.EnableEurekaServer;
@SpringBootApplication
@EnableEurekaServer
public class EurekaServerApplication {
public static void main(String[] args) {SpringApplication.run(EurekaServerApplication.class, args);}
}

启动Eureka服务器和DemoServiceApplication,访问Eureka控制台(通常是http://localhost:8761),应该能够看到demo-service已经注册成功。访问http://localhost:<port>/hello<port>demo-service的端口),应该能够看到返回的“Hello from Demo Service!”。

现在,当你启动或停止demo-service实例时,Eureka服务器会实时地感知到这一变化,并更新其注册列表。其他微服务可以通过Eureka服务器查询最新的服务地址信息,从而实现动态的服务发现和调用。

Python 演示

以下是一个使用Python和Consul实现微服务上下线动态感知的示例。

安装Consul客户端

首先,需要安装python-consul库:

bash复制代码
pip install python-consul
注册服务
python复制代码
import consul
import time
import random
# 连接到Consul代理
c = consul.Consul(host='127.0.0.1', port=8500)
# 注册服务
def register_service(name, port):c.agent.service.register(name, service_id=f"{name}_{port}", address='127.0.0.1', port=port, check={
'ttl': '10s'})
# 模拟服务运行
def run_service(name, port):register_service(name, port)
try:
while True:
print(f"{name} is running on port {port}")time.sleep(random.randint(1, 10))
except KeyboardInterrupt:deregister_service(name, port)
# 注销服务
def deregister_service(name, port):c.agent.service.deregister(f"{name}_{port}")
if __name__ == "__main__":service_name = "demo-service"service_port = 8080run_service(service_name, service_port)
发现和调用服务
python复制代码
import consul
import requests
# 连接到Consul代理
c = consul.Consul(host='127.0.0.1', port=8500)
# 发现服务
def discover_services(service_name):index, services = c.catalog.service(service_name)
return [(service['Address'], service['ServicePort']) for service in services]
# 调用服务
def call_service(service_name):services = discover_services(service_name)
if not services:
print(f"No available instances of {service_name}")
returnaddress, port = random.choice(services)url = f"http://{address}:{port}/hello"response = requests.get(url)
print(f"Called {url}, response: {response.text}")
if __name__ == "__main__":service_name = "demo-service"call_service(service_name)

在上面的示例中,我们首先注册了一个名为demo-service的服务,并模拟了其运行过程。然后,我们编写了一个发现服务的函数discover_services,用于查询Consul中的服务实例。最后,我们编写了一个调用服务的函数call_service,它随机选择一个可用的服务实例并发送HTTP GET请求。

结论

微服务上下线动态感知是微服务架构中一个非常重要的功能,它确保了系统的可用性和负载均衡。通过服务注册与发现机制、心跳机制、事件通知机制以及优雅上下线策略的实现,我们可以有效地感知和处理服务的上下线变化。同时,结合Zookeeper、Spring Cloud与Eureka、Consul等技术和工具,我们可以更加灵活地构建和管理微服务系统。在未来的发展中,随着技术的不断进步和应用场景的不断拓展,微服务上下线动态感知的实现方式也将更加多样化和智能化。

相关文章:

微服务上下线动态感知实现的技术解析

序言 随着微服务架构的广泛应用&#xff0c;服务的动态管理和监控变得尤为重要。在微服务架构中&#xff0c;服务的上下线是一个常见的操作&#xff0c;如何实时感知这些变化&#xff0c;确保系统的稳定性和可靠性&#xff0c;成为了一个关键技术挑战。本文将深入探讨微服务上…...

指针与引用错题汇总

int *p[3]; // 定义一个包含 3 个指向 int 的指针的数组int a 10, b 20, c 30; p[0] &a; // p[0] 指向 a p[1] &b; // p[1] 指向 b p[2] &c; // p[2] 指向 c // 访问指针所指向的值 printf("%d %d %d\n", *p[0], *p[1], *p[2]); // 输出: 10 20 30…...

短视频账号矩阵系统源码--独立saas技术部署

短视频矩阵系统通过多账号在多个平台上发布内容&#xff0c;形成一种网络效应。对于抖音平台而言&#xff0c;技术公司需具备特定接口权限方能进行开发工作。然而&#xff0c;视频发布及企业号评论与回复等功能的接口权限往往难以获取。通过构建抖音账号矩阵&#xff0c;利用多…...

leaflet 介绍

目录 一、leaflet 官网 二、leaflet 在项目中的引用 1、在head中引入 2、在main.js中引入 leaflet目前版本是1.9.4&#xff0c;在leaflet插件库中&#xff0c;很多插件因长时间未更新&#xff0c;适配的是1.7版本的&#xff0c;在选用插件的时候要查看版本适配。 leaflet详…...

总结贴:Servlet过滤器、MVC拦截器

一:Servlet过滤器 1.1解析 Filter 即为过滤&#xff0c;用于请求到达Servlet之前(Request),以及再Servlet方法执行完之后返回客户端进行后处理(HttpServletResponse)。简单说就是对请求进行预处理&#xff0c;对响应进行后处理 在请求到达Servlet之前,可以经过多个Filt…...

鸿蒙开发:自定义一个任意位置弹出的Dialog

前言 鸿蒙开发中&#xff0c;一直有个问题困扰着自己&#xff0c;想必也困扰着大多数开发者&#xff0c;那就是&#xff0c;系统提供的dialog自定义弹窗&#xff0c;无法实现在任意位置进行弹出&#xff0c;仅限于CustomDialog和Component struct的成员变量&#xff0c;这就导致…...

在Windows下编译支持https的wsdl2h

下载源码 在官网下载源码 安装Openssl 下载OpenSSL并安装&#xff0c;安装完成后需要将OpenSSL的路径添加到环境变量中 配置VS 1、打开工程 2、因为前面安装的OpenSLL是64位的&#xff0c;因此需要创建一个X64的配置 打开配置管理器&#xff0c;然后选择新建&#xff0…...

PHP和GD库如何根据像素绘制图形

使用PHP和GD库&#xff0c;你可以根据像素绘制各种图形&#xff0c;比如点、线、矩形、圆形等。GD库是PHP的一个扩展&#xff0c;它提供了一系列用于创建和处理图像的函数。以下是一个简单的示例&#xff0c;展示如何使用GD库根据像素绘制图形。 安装GD库 首先&#xff0c;确…...

webpack(react)基本构建

文章目录 概要整体架构流程技术名词解释技术细节小结 概要 Webpack 是一个现代 JavaScript 应用程序的静态模块打包工具。它的主要功能是将各种资源&#xff08;如 JavaScript、CSS、图片等&#xff09;视为模块&#xff0c;并将它们打包成一个或多个输出文件&#xff0c;以便…...

《Opencv》基础操作<1>

目录 一、Opencv简介 主要特点&#xff1a; 应用领域&#xff1a; 二、基础操作 1、模块导入 2、图片的读取和显示 &#xff08;1&#xff09;、读取 &#xff08;2&#xff09;、显示 3、 图片的保存 4、获取图像的基本属性 5、图像转灰度图 6、图像的截取 7、图…...

Oracle 11g R2 RAC 到单实例 Data Guard 搭建(RMAN备份方式)

一、配置方案 环境说明 角色主库主库备库主机名rac01rac02racdg公网IP10.10.10.14110.10.10.14310.10.10.191VIP10.10.10.14210.10.10.144-SCAN10.10.10.14010.10.10.140-INSTANCE_NAMEorcl1orcl2orclDB_NAMEorclorclorclSERVICE_NAMEorclorclorclDB_UNIQUE_NAMEorclorclorcl…...

HTTPS 加密

HTTPS 加密技术 1. HTTPS 概述 HTTPS&#xff08;HyperText Transfer Protocol Secure&#xff09;是 HTTP 协议的安全版本&#xff0c;利用 SSL/TLS 协议对通信进行加密&#xff0c;确保数据的机密性、完整性和身份认证。HTTPS 在保护敏感数据的传输&#xff08;如登录凭证、…...

泛微e9开发 编写前端请求后端接口方法以及编写后端接口

泛微e9开发 前端请求后端接口以及后端发布接口 前端请求后端接口 前端发起get请求 fetch(/api/youpath, {method: GET, // 默认 GET 方法&#xff0c;可以省略headers: {Content-Type: application/json, // 通常 GET 请求无需指定 body&#xff0c;Content-Type 不太重要},…...

Linux —— 《线程控制》

文章目录 前言&#xff1a;为什么要链接pthread库&#xff1f;线程控制&#xff1a;线程创建&#xff1a;start_routine?传递自定义类型同一份栈空间&#xff1f; 线程等待&#xff1a;返回值与参数&#xff1f;创建多线程 线程终止线程分离 前言&#xff1a; 上一文我们学习…...

基于HTML+CSS的房地产销售网站设计与实现

摘 要 房地产销售系统&#xff0c;在二十年来互联网时代下有着巨大的意义&#xff0c;随着互联网不断的发展扩大&#xff0c;一个方便直 观的房地产管理系统的网站开发是多么地有意义&#xff0c;不仅打破了传统的线下看房&#xff0c;线下获取资讯&#xff0c;也给房地产从业…...

操作系统 | 学习笔记 | 王道 | 2.4死锁

2.4 死锁 文章目录 2.4 死锁2.4.1 死锁的概念2.4.2 死锁预防2.4.3 死锁避免2.4.4 死锁检测和解除 2.4.1 死锁的概念 死锁的定义 在并发环境下&#xff0c;各进程因竞争资源而造成的一种互相等待对方手里的资源&#xff0c;导致各进程都阻塞&#xff0c;都无法向前推进的现象&am…...

【FPGA开发】Vivado自定义封装IP核,绑定总线

支持单个文件的封装、整个工程的封装&#xff0c;这里用单个文件举例。 在文件工程目录下&#xff0c;自建一个文件夹&#xff0c;里面放上需要封装的verilog文件。 选择第三个&#xff0c;指定路径封装&#xff0c;找到文件所在目录 取个名&#xff0c;选择封装IP的路径 会…...

python的3D可视化库vedo-3 (visual模块)点对象的属性、光效、附注

文章目录 3 PointsVisual的方法3.1 对象属性3.1.1 顶点大小3.1.2 复制属性3.1.3 颜色设置3.1.4透明度设置 3.2 对象光效3.2.1 点的形状3.2.2 点的表面光效 3.3 尾随线和投影3.3.1 尾随线3.3.2 投影 3.4 给对象附加文字说明3.4.1 标注3.4.2 2D标注3.4.3 气泡说明3.4.4 旗标说明3…...

llamaindex实战-ChatEngine-ReAct Agent模式

概述 ReAct 是一种基于Agent的聊天模式&#xff0c;构建在数据查询引擎之上。对于每次聊天交互&#xff0c;代理都会进入一个 ReAct 循环&#xff1a; 首先决定是否使用查询引擎工具并提出适当的输入 &#xff08;可选&#xff09;使用查询引擎工具并观察其输出 决定是否重复…...

redis快速进门

、数据库类型认识 关系型数据库 关系型数据库是一个结构化的数据库&#xff0c;创建在关系模型&#xff08;二维表格模型&#xff09;基础上&#xff0c;一般面向于记录。 SQL 语句&#xff08;标准数据查询语言&#xff09;就是一种基于关系型数据库的语言&#xff0c;用于执行…...

Mybatis逆向工程,动态创建实体类、条件扩展类、Mapper接口、Mapper.xml映射文件

今天呢&#xff0c;博主的学习进度也是步入了Java Mybatis 框架&#xff0c;目前正在逐步杨帆旗航。 那么接下来就给大家出一期有关 Mybatis 逆向工程的教学&#xff0c;希望能对大家有所帮助&#xff0c;也特别欢迎大家指点不足之处&#xff0c;小生很乐意接受正确的建议&…...

ESP32 I2S音频总线学习笔记(四): INMP441采集音频并实时播放

简介 前面两期文章我们介绍了I2S的读取和写入&#xff0c;一个是通过INMP441麦克风模块采集音频&#xff0c;一个是通过PCM5102A模块播放音频&#xff0c;那如果我们将两者结合起来&#xff0c;将麦克风采集到的音频通过PCM5102A播放&#xff0c;是不是就可以做一个扩音器了呢…...

Nginx server_name 配置说明

Nginx 是一个高性能的反向代理和负载均衡服务器&#xff0c;其核心配置之一是 server 块中的 server_name 指令。server_name 决定了 Nginx 如何根据客户端请求的 Host 头匹配对应的虚拟主机&#xff08;Virtual Host&#xff09;。 1. 简介 Nginx 使用 server_name 指令来确定…...

图表类系列各种样式PPT模版分享

图标图表系列PPT模版&#xff0c;柱状图PPT模版&#xff0c;线状图PPT模版&#xff0c;折线图PPT模版&#xff0c;饼状图PPT模版&#xff0c;雷达图PPT模版&#xff0c;树状图PPT模版 图表类系列各种样式PPT模版分享&#xff1a;图表系列PPT模板https://pan.quark.cn/s/20d40aa…...

精益数据分析(97/126):邮件营销与用户参与度的关键指标优化指南

精益数据分析&#xff08;97/126&#xff09;&#xff1a;邮件营销与用户参与度的关键指标优化指南 在数字化营销时代&#xff0c;邮件列表效度、用户参与度和网站性能等指标往往决定着创业公司的增长成败。今天&#xff0c;我们将深入解析邮件打开率、网站可用性、页面参与时…...

Springboot社区养老保险系统小程序

一、前言 随着我国经济迅速发展&#xff0c;人们对手机的需求越来越大&#xff0c;各种手机软件也都在被广泛应用&#xff0c;但是对于手机进行数据信息管理&#xff0c;对于手机的各种软件也是备受用户的喜爱&#xff0c;社区养老保险系统小程序被用户普遍使用&#xff0c;为方…...

React---day11

14.4 react-redux第三方库 提供connect、thunk之类的函数 以获取一个banner数据为例子 store&#xff1a; 我们在使用异步的时候理应是要使用中间件的&#xff0c;但是configureStore 已经自动集成了 redux-thunk&#xff0c;注意action里面要返回函数 import { configureS…...

在QWebEngineView上实现鼠标、触摸等事件捕获的解决方案

这个问题我看其他博主也写了&#xff0c;要么要会员、要么写的乱七八糟。这里我整理一下&#xff0c;把问题说清楚并且给出代码&#xff0c;拿去用就行&#xff0c;照着葫芦画瓢。 问题 在继承QWebEngineView后&#xff0c;重写mousePressEvent或event函数无法捕获鼠标按下事…...

动态 Web 开发技术入门篇

一、HTTP 协议核心 1.1 HTTP 基础 协议全称 &#xff1a;HyperText Transfer Protocol&#xff08;超文本传输协议&#xff09; 默认端口 &#xff1a;HTTP 使用 80 端口&#xff0c;HTTPS 使用 443 端口。 请求方法 &#xff1a; GET &#xff1a;用于获取资源&#xff0c;…...

Netty从入门到进阶(二)

二、Netty入门 1. 概述 1.1 Netty是什么 Netty is an asynchronous event-driven network application framework for rapid development of maintainable high performance protocol servers & clients. Netty是一个异步的、基于事件驱动的网络应用框架&#xff0c;用于…...