大部分项目部署中,为了方便,可能都直接使用kill -9 服务的pid来停掉服务。
但是由于Eureka采用心跳的机制来上下线服务,会导致服务消费者调用此已经kill的服务提供者然后出错。
可以采用以下方式来解决:
核心是先调用方法主动通知Eureka注册中心服务下线,然后再停掉服务。

本文会介绍几种eureka 注册中心服务下线的方式

最不可取的就是直接使用kill命令停掉服务。
默认情况下,如果Eureka Server在90秒没有收到Eureka客户的续约,它会将实例从其注册表中删除。但这种做法的不好之处在于, 客户端已经停止了运行,但仍然在注册中心的列表中。 虽然通过一定的负载均衡策略或使用熔断器可以让服务正常进行,但有没有方法让注册中心马上知道服务已经下线呢?

1、通过/offline请求来实现服务下线
在启动eureka服务的时候发现控制台有以下的输出
在这里插入图片描述

由此猜想可以通过改接口下线服务, 于是尝试了一下
在这里插入图片描述

果然能从注册中心中移除该实例
在这里插入图片描述

2.向eureka 注册中心发送delete 请求

格式为 /eureka/apps/{application.name}/

下面是下线一个hello-service的例子。
在这里插入图片描述

下图是用postman 发送delete请求

在这里插入图片描述

3. 客户端主动通知注册中心下线

如果你的eureka客户端是是一个spring boot应用,可以通过调用以下代码通知注册中心下线。

DiscoveryManager.getInstance().shutdownComponent();

例子如下,

@RestController
public class HelloController {
    @Autowired
    private DiscoveryClient client;
 
    @RequestMapping(value = "/hello", method = RequestMethod.GET)
    public String index() {
        java.util.List<ServiceInstance> instances = client.getInstances("hello-service");       
        return "Hello World";
    }
    
    @RequestMapping(value = "/offline", method = RequestMethod.GET)
    public void offLine(){
    	DiscoveryManager.getInstance().shutdownComponent();
    }   
}

4、在服务器上用curl发送post请求到pause.

curl -X POST http://localhost:8080/pause
此时eurake上该服务被标记问下线,但该服务其实还是可以正常访问的,当client还未及时更新本地Instances缓存时,依然不会中断服务。当所有client都感知到该服务DOWN后就不会再往该服务发请求了。

5、通过actuator 监控组件来实现优雅停机*
(1)、在微服务pom.xml文件中,配置spring-boot-starter-actuator 监控组件

org.springframework.boot spring-boot-starter-actuator

(2)、application.yml配置

#管理端点

management: 
#  endpoints:
#    web:
#      base-path: /actuator  #默认为 /actuator
  endpoints: 
    web:
      exposure:
        include:
        - shutdown
        - info
        - health
  endpoint:
    shutdown:
      enabled: true

#配置management的自定义端口,可以与server.port不同,
#management.server.port: 8081
#management.server.address: 127.0.0.1 # management只能在本机访问

(3)、在任意一台服务器上利用curl发送shutdown命令

curl -X POST http://ip:端口/actuator/shutdown

值得注意的是,Eureka客户端每隔一段时间(默认30秒)会发送一次心跳到注册中心续约。如果通过这种方式下线了一个服务,而没有及时停掉的话,该服务很快又会回到服务列表中。
所以,可以先停掉服务,再发送请求将其从列表中移除。

以上就是针对spring cloud微服务优雅下线的方法的总结,个人最推荐通过actuator 监控组件来实现。
其他方式有待验证。

Logo

权威|前沿|技术|干货|国内首个API全生命周期开发者社区

更多推荐