赞
踩
大部分项目部署中,为了方便,可能都直接使用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 监控组件
(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 监控组件来实现。
其他方式有待验证。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。