一、介绍 微服务架构中的应用优雅停机主要是指应用实例有计划而平滑(即不产生须要处理的事故)的退出。应用服务器的停机主要分为两类:主动停机和被动停机,而其中主动停机和大部分的被动停机都是能够实现优雅停机。若是应用不作优雅停机,则会带来如下状况:后端
•数据丢失:内存的中数据还没有持久化至磁盘 •文件损坏:正在操做写的文件因没有更新完成,致使文件损坏 •请求丢失:排队中等待处理的请求丢失 •响应丢失:成功的交易还没来得及作出响应 •交易中断:正在处理至中间状态的交易被强制中断 •服务未下线:上游服务依然还会继续往下游服务发送消费请求 而咱们微服务的优雅升级的目标就是避免以上几种状况,从而避免人工干预的工做量和提高微服务架构的服务高可靠。服务器
二、使用场景 优雅停机能够解决如下场景:架构
•KILL PID •应用意外自动退出 •使用脚本命令的方式中止应用 •优雅停机解决不了如下场景:ide
•忽然断电 •机器物理破坏 •KILL-9 PID 或 taskkill /f /pid微服务
三、ShutdownHook Java的优雅停机一般经过注册JDK的ShutdownHook(钩子)来实现,当系统接收到退出指令后,首先标记系统处于退出状态,再也不接收新的消息,而后将积压的消息处理完,最后调用资源回收接口将资源销毁,最后各线程退出执行。简单的使用demo案例以下(简单版):线程
/**设计
优雅停机处理方式cdn
@author lry **/ public class Main{blog
/**接口
启动应用 **/ public void start(){ // 第一步:启动应用服务……
// 第二步:注册JDK钩子 Runtime.getRuntime().addShutdownHook(new Thread(new Runnable() { @Override public void run() { System.out.println(“The hook running…”); //第三步:调用停机处理 stop(); } })); }
/**
中止应用 **/ public void stop(){ // 中止应用前停机处理(如:注销服务、标记不接受请求等) } } 超时控制 一般优雅退出须要有超时控制机制,若是到达超时时间仍然没有完成退出前的资源回收等操做,则由停机脚本直接调用KILL -9 PID的方式进行强制退出,否则可能会等待很长时间。
四、微服务优雅停机 微服务的优雅停机没有统一的解决方案,只要抓住核心思想进行设计便可: 引流 → 挡板 → 等待停机
但在微服务架构中,咱们能够遵照如下建议规则来设计微服务的优雅停机机制: •全部微服务应用都应该支持优雅停机 •优先注销注册中心注册的服务实例 •待停机的服务应用的接入点标记拒绝服务 •上游服务支持故障转移因优雅停机而拒绝的服务 •根据具体业务也提供适当的停机接口 微服务应用的优雅停机根据其使用者角色的不一样,而主要分为两种类型:
•微服务业务应用优雅停机设计:
其他各层设备的优雅停机均可从以上两种类型进行衍生出解决方案,如:
•整个后端架构升级,则可从DNS或Nginx直接切换 •Nginx层升级,则能够从DNS直接切换
五、使用案例 在业界开源的产品中,不少产品都使用了JDK钩子的方式来实现优雅停机,如如下产品:
•Netty •DUBBO