SpringCloud中微服务的优雅停机

/ Java / 没有评论 / 2278浏览

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

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

最不可取的就是直接使用kill命令停掉服务。

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

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

格式为/eureka/apps/{application.name}/,下面是下线一个hello-service的例子。 下图是用postman 发送delete请求

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

如果你的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();
    }   
}

在启动eureka服务的时候发现控制台有以下的输出 由此猜想可以通过改接口下线服务, 于是尝试了一下 果然能从注册中心中移除该实例

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

curl -X POST http://localhost:8080/pause

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

4、通过actuator 监控组件来实现优雅停机*

(1)、在微服务pom.xml文件中,配置spring-boot-starter-actuator 监控组件

<dependencies>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-actuator</artifactId>
    </dependency>
</dependencies>

(2)、application.yml配置

# 管理端点
management: 
  endpoints: 
    web:
      base-path: /actuator  #默认为 /actuator
      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 监控组件来实现。其他方式有待验证。