老板:kill -9 的原理都不知道就敢去线上执行?明天不用来了
作者:HollisChuang
相信很多程序员对于Linux系统都不陌生,即使自己的日常开发机器不是Linux,那么线上服务器也大部分都是的,所以,掌握常用的Linux命令也是程序员必备的技能。
但是,怕就怕很多人对于部分命令只是一知半解,使用不当就能导致线上故障。
前段时间,我们的线上应用报警,频繁FGC,需要紧急处理问题,于是有同事去线上重启机器(正常程序应该是先采集堆dump,然后再重启,方便排查是否存在内存泄露等问题)。
但是在重启过程中,同事发现正常的重启命令应用无反应,然后尝试使用kill命令"杀"掉Java进程,但是仍然无效。于是他私自决定使用 "kill -9"结束了进程的生命。
虽然应用进程被干掉了,但是随之而来带来了很多问题,首先是上游系统突然发生大量报警,对应开发找过来说调用我们的RPC服务无响应,频繁超时。
后来,我们又发现系统中存在部分脏数据,有些在同一个事务中需要完整更新的数据,只跟新了一半...
为什么正常的kill无法"杀掉"进程,而kill -9就可以?为什么kill -9会引发这一连串连锁反应?正常的kill执行时,JVM会如何处理的呢?
要搞清楚这些问题,我们要先从kill命令说起。
kill 命令
我们都知道,想要在Linux中终止一个进程有两种方式,如果是前台进程可以使用Ctrl+C键进行终止;如果是后台进程,那么需要使用kill命令来终止。(其实Ctrl+C也是kill命令)
kill命令的格式是:
kill[参数][进程号]如:
kill 21121
kill -9 21121
其中[参数]是可选的,进程号可以通过jps/ps/pidof/pstree/top等工具获取。
kill的命令参数有以下几种:
-l 信号,若果不加信号的编号参数,则使用“-l”参数会列出全部的信号名称
-a 当处理当前进程时,不限制命令名和进程号的对应关系
-p 指定kill 命令只打印相关进程的进程号,而不发送任何信号
-s 指定发送信号
-u 指定用户
通常情况下,我们使用的-l(信号)的时候比较多,如我们前文提到的kill -9中的9就是信号。
信号如果没有指定的话,默认会发出终止信号(15)。常用的信号如下:
HUP 1 终端断线
INT 2 中断(同 Ctrl + C)
QUIT 3 退出(同 Ctrl + \)
TERM 15 终止
KILL 9 强制终止
CONT 18 继续(与STOP相反, fg/bg命令)
STOP 19 暂停(同 Ctrl + Z)
比较常用的就是强制终止信号:9和终止信号:15,另外,中断信号:2其实就是我们前文提到的Ctrl + C结束前台进程。
那么,kill -9 和 kill -15到底有什么区别呢?该如何选择呢?
kill -9 和 kill -15的区别
kill命令默认的信号就是15,首先来说一下这个默认的kill -15信号。
当使用kill -15时,系统会发送一个SIGTERM的信号给对应的程序。当程序接收到该信号后,具体要如何处理是自己可以决定的。
这时候,应用程序可以选择:
- 1、立即停止程序
- 2、释放响应资源后停止程序
- 3、忽略该信号,继续执行程序
因为kill -15信号只是通知对应的进程要进行"安全、干净的退出",程序接到信号之后,退出前一般会进行一些"准备工作",如资源释放、临时文件清理等等,如果准备工作做完了,再进行程序的终止。
但是,如果在"准备工作"进行过程中,遇到阻塞或者其他问题导致无法成功,那么应用程序可以选择忽略该终止信号。
这也就是为什么我们有的时候使用kill命令是没办法"杀死"应用的原因,因为默认的kill信号是SIGTERM(15),而SIGTERM(15)的信号是可以被阻塞和忽略的。
和kill -15相比,kill -9就相对强硬一点,系统会发出SIGKILL信号,他要求接收到该信号的程序应该立即结束运行,不能被阻塞或者忽略。
所以,相比于kill -15命令,kill -9在执行时,应用程序是没有时间进行"准备工作"的,所以这通常会带来一些副作用,数据丢失或者终端无法恢复到正常状态等。
Java是如何处理SIGTERM(15)的
我们都知道,在Linux中,Java应用是作为一个独立进程运行的,Java程序的终止运行是基于JVM的关闭实现的,JVM关闭方式分为3种:
正常关闭:当最后一个非守护线程结束或者调用了System.exit或者通过其他特定平台的方法关闭(接收到SIGINT(2)、SIGTERM(15)信号等)
强制关闭:通过调用Runtime.halt方法或者是在操作系统中强制kill(接收到SIGKILL(9)信号)
异常关闭:运行中遇到RuntimeException异常等。
JVM进程在接收到kill -15信号通知的时候,是可以做一些清理动作的,比如删除临时文件等。
当然,开发者也是可以自定义做一些额外的事情的,比如让tomcat容器停止,让dubbo服务下线等。
而这种自定义JVM清理动作的方式,是通过JDK中提供的shutdown hook实现的。JDK提供了
Java.Runtime.addShutdownHook(Thread hook)方法,可以注册一个JVM关闭的钩子。
例子如下:
package com.hollis;public class ShutdownHookTest {public static void main(String[] args) {boolean flag = true;Runtime.getRuntime().addShutdownHook(new Thread(() -> {System.out.println("hook execute...");}));while (flag) {// app is runing}System.out.println("main thread execute end...");}
}
执行命令:
➜ jps
6520 ShutdownHookTest
6521 Jps
➜ kill 6520
控制台输出内容:
hook execute...
Process finished with exit code 143 (interrupted by signal 15: SIGTERM)
可以看到,当我们使用kill(默认kill -15)关闭进程的时候,程序会先执行我注册的shutdownHook,然后再退出,并且会给出一个提示:interrupted by signal 15: SIGTERM
如果我们执行命令kill -9:
➜ kill -9 6520
控制台输出内容:
Process finished with exit code 137 (interrupted by signal 9: SIGKILL)
可以看到,当我们使用kill -9 强制关闭进程的时候,程序并没有执行shutdownHook,而是直接退出了,并且会给出一个提示:interrupted by signal 9: SIGKILL
总结
kill命令用于终止Linux进程,默认情况下,如果不指定信号,kill 等价于kill -15。
kill -15执行时,系统向对应的程序发送SIGTERM(15)信号,该信号是可以被执行、阻塞和忽略的,所以应用程序接收到信号后,可以做一些准备工作,再进行程序终止。
有的时候,kill -15无法终止程序,因为他可能被忽略,这时候可以使用kill -9,系统会发出SIGKILL(9)信号,该信号不允许忽略和阻塞,所以应用程序会立即终止。
这也会带来很多副作用,如数据丢失等,所以,在非必要时,不要使用kill -9命令,尤其是那些web应用、提供RPC服务、执行定时任务、包含长事务等应用中,因为kill -9 没给spring容器、tomcat服务器、dubbo服务、流程引擎、状态机等足够的时间进行收尾。
最后,很多人会说,说了这么多,不是还得用 kill -9 吗?
其实,本文的目的不是不让大家用,那就是因噎废食了。本文是希望大家可以了解其背后的原理,知道他可能带来的副作用。在选择要不要执行的时候,可以考虑到这些因素,如果能够针对可能发生的副作用,提前做好预案和心理准备,然后再执行,那就很完美了。
在执行之后,发生了非预期的问题时,大家可以想到有可能和kill -9有关,那本文的目的也算达到了。
来源:掘金 链接:
https://juejin.im/post/5ebb5273e51d454dc87f2f60
老板:kill -9 的原理都不知道就敢去线上执行?明天不用来了相关推荐
- 老板:kill -9 的原理都不知道就敢去线上执行?明天不用来了!
GitHub 14.5k Star 的Java工程师成神之路,开放阅读了! 相信很多程序员对于Linux系统都不陌生,即使自己的日常开发机器不是Linux,那么线上服务器也大部分都是的,所以,掌握常用 ...
- 老板:kill -9 的原理都不知道就敢在线上执行?
作者 l Hollis 来源 l Hollis(ID:hollischuang) 相信很多程序员对于Linux系统都不陌生,即使自己的日常开发机器不是Linux,那么线上服务器也大部分都是的,所以,掌 ...
- 老板:kill -9的原理都不知道就敢到线上执行,明天不用来了
点击上方蓝色"方志朋",选择"设为星标" 回复"666"获取独家整理的学习资料! 这是Hollis的第 270篇原创分享 作者 l Holl ...
- 老板:kill -9的原理都不知道就敢到线上执行,明天不用来了!
相信很多程序员对于Linux系统都不陌生,即使自己的日常开发机器不是Linux,那么线上服务器也大部分都是的,所以,掌握常用的Linux命令也是程序员必备的技能. 但是,怕就怕很多人对于部分命令只是一 ...
- 不要给我说什么底层原理_连集合底层实现原理都不知道,你敢说 Redis 用的很溜?...
目录 SDS 的设计到底有多牛逼. List.Set.Sorted Set.Hash 底层实现原理 SDS 的设计到底有多牛逼 Redis 使用 C 语言编写,但是并没有直接使用 C 语言自带的字符串 ...
- 老弟,你连HTTPS 原理都不懂,还给我讲“中间人攻击”,逗我吗...
点击上方"方志朋",选择"设为星标" 回复"666"获取新整理的面试资料 作者:leapmie 来源:https://urlify.cn/z ...
- 最近面试了一位4年的Java,什么都不知道!还自认为很牛逼...
最近看了很多简历,很多候选人年限不小,但是做的都是一些非常传统的项目,想着也不能通过简历就直接否定一个人,何况现在大环境越来 越难,大家找工作也不容易,于是就打算见一见. 在沟通中发现,由于年限不小, ...
- 考研350什么水平计算机,考研350分的难度相当于高考考什么水平?很多人都不知道...
原标题:考研350分的难度相当于高考考什么水平?很多人都不知道 很多参加考研的同学可能知道总分是500分,但是却都不大懂考研高分的概念,下面,和高考分数作对比,大家都可以听听几个网友怎么说的 网友一: ...
- 这类注解都不知道,还好意思说用过Spring Boot?
持续原创输出,点击上方蓝字关注我 目录 前言 Spring Boot 版本 @Conditional Condition接口 ConditionContext接口 如何自定义Condition? 条件 ...
最新文章
- 多目标跟踪:CVPR2019论文阅读
- div高度、宽度100% div width、height 100%
- layoutSubviews什么时候触发调用
- [渝粤教育] 西南科技大学 电器设备及维修 在线考试复习资料
- java8的路径_什么是路径?
- linux使用外部smtp发送邮件,Centos 配置mailx使用外部smtp发送邮件
- 字符串字母大小写转换
- 项目中用到的ws2811炫彩灯控制程序
- 【深度学习】2.1深度学习的实用层面
- c语言常用颜色种类,C语言常用颜色种类(国外英语资料).doc
- 使用face-api.js人脸识别让照片中的人物自动流汗(自动流汗黄豆/差不多得了)
- 腾讯自研HIDS「洋葱」后台上云架构演进实践
- android2012系统,压倒性份额四核技术 Android系统2012前瞻
- android 左测设置功能,iqoo手机左边按键有什么作用?iqoo手机左侧按键介绍
- 微信小程序手把手教你实现类似Android中ViewPager控件效果
- 图数据库nebula
- 敏捷回顾会:经验教训的总结
- 小鲨鱼在51nod小学
- [转载]STL之priority_queue_彭世瑜_新浪博客
- 《约会专家》片尾【约会宝典】总结