神奇的Java僵尸(defunct)进程问题排查过程
现象描述
大概1个月多以前 在启动脚本中增加了tail -f
用来启动后追踪日志判断是否启动成功
后发现无法执行shutdown.sh(卡住 利用curl) 然后无奈使用kill -9
但通过ps -el 发现此时进程变为defunct 即僵尸进程
当时的解决办法无奈 只能找到僵尸进程的父进程kill
当时认为可能是tail的问题 后来启动脚本中去掉tail 发现问题解决
But
当时一直没有来得及排查是如何引起僵尸进程的问题
这两天抽时间排查了一下 发现和tail没有一毛钱关系
艰难的排查过程1-尝试复现
测试代码Defunct.java

启动脚本start.sh

启动脚本start_tail.sh 使用了tail

关服脚本stop.sh 这里使用kill关服

分别用两个脚本测试,得出下面几个结论:
所以初步结论是貌似和tail没有什么关系
此时sh和tail两个进程都结束了
而此时java进程的父进程变为了1
sh分别有两个子进程
一个是java子进程 一个是tail子进程
start.sh启动的java进程的父进程是1 即init进程
start_tail.sh启动后 java进程的父进程是sh
当启动start_tail.sh后 因为tail是前台进程 所以ctrl+c可以结束
用这个例子做各种测试 都无法复现僵尸进程的问题
艰难的排查过程2-游戏服务器尝试复现
当初出现是在游戏服务器复现的 那么应该比较复现吧
修改了一个游戏服务器的启动脚本 默认是没有加tail 现在加上了tail -f
启动游戏服务器脚本 看到日志 启动成功 ctrl+c 退出tail
调用shutdown.sh 发现服务器顺利关闭
结论:竟然无法在游戏服务器复现
艰难的排查过程3-各种思考、查阅资料
首先从僵尸进程的产生原因入手,猜测是否是sh这个父进程没有调用waitpid去回收java子进程。
查询网上类似的tomcat tail -f问题,思考当初1个多月以前的情形,其中有一个很重要的当初情形是shutdown的时候ctrl卡住了。灵光一现,难道是当初操作失误了,没有按下ctrl+c而是按下了ctrl+z。
神奇的ctrl+z 复现测试代码defunct
启动start_tail.sh 然后ctrl+z

启动stop.sh 发现进程(3974)无法被stop

使用kill -9 尝试杀死进程 此时发现进程已经是defunct了

此时只要使用fg命令从后台调到前台然后按下ctrl+c 则僵尸进程自动消失

神奇的ctrl+z 复现游戏服务器defunct
启动脚本(有tail) 等待一段时间(将所有服务器全部开启) 并ctrl+z

此时执行shutdown.sh 发现没有任何反应(卡住) 无奈ctrl+c

此时执行jstack 也发现没有任何反应(卡住) 无奈ctrl+c

此时执行kill -9 此时java进程已经变为了僵尸进程

此时用fg将暂停的脚本恢复 然后ctrl+c 则僵尸进程消失 顺利被回收

总结1
tail和造成defunct没有任何关系
根本原因是因为按下ctrl+z 将start_tail.sh切换到了后台
测试1 当start_tail.sh后 按下ctrl+z 如果直接被crt#session关闭了呢
更神奇的事情发生了 java进程直接被干掉了
!!这个在游戏服务器也测试了 一定要注意!!
测试2 执行start_tail.sh 直接关闭ctr#session 则java进程还在 因为是nohup启动
测试3 当start_tail.sh后 按下ctrl+z 再按fg 恢复执行 此时之后可以顺利shutdown
总结2
正常启动脚本 没有tail java进程的父进程是1 即init进程 使用shutdown脚本关闭java进程后 自动被init进程回收
启动脚本加了tail
此时java进程的父进程是sh进程
sh进程有两个子进程 一个是java子进程 一个是tail子进程
直接ctrl+c 则sh进程和tail进程都结束 java进程的父进程变为了1
如果不ctrl+c 直接shutdown java进程 则java进程也会正常结束,即sh父进程会回收java子进程
总结3
最终'罪魁祸首'是ctrl+z ,它会暂停程序的运行
如果我们启动脚本没有加tail 则执行完nohup & 自动到后台
但是我们加了tail后 因为tail是前台进程 所以要么ctrl+c结束 要么ctrl+z
如果我们按下了ctrl+z 则sh启动的所有子进程都会暂停
所以我们的java进程此时处于暂停状态 所以shutdown/jstack都卡住了一样 只能ctrl+c退出
然后错误的操作就是使用kill -9 这个会把进程给干掉 但是因为父进程sh被暂停了 所以无法waitPid 执行子进程的回收操作 从而导致java进程变为了僵尸进程
而通过fg恢复后 ctrl+c 父进程和tail都退出 java进程被init进程接管 自动回收
总结4
加tail -f 没有问题,但是一定不要忘了ctrl+c;
如果ctrl+z 那么一定要fg然后ctrl+c;
当出现shutdown.sh卡住或者操作jvm都没反应,则可以怀疑是暂停引起的;
————————————
原文链接:https://www.jianshu.com/u/21add3dce532
神奇的Java僵尸(defunct)进程问题排查过程的更多相关文章
- 僵尸进程的产生和避免,如何kill杀掉linux系统中的僵尸defunct进程
在 Unix系统管理中,当用ps命令观察进程的执行状态时,经常看到某些进程的状态栏为defunct,这就是所谓的"僵尸"进程."僵尸"进程是一个早已 死亡的进程 ...
- 【转】Defunct进程 僵尸进程
在测试基于 DirectFB+Gstreamer 的视频联播系统的一个 Demo 的时候,其中大量使用 system 调用的语句,例如在 menu 代码中的 system("./play&q ...
- <转>杜绝 Defunct进程 僵尸进程
http://hanover.iteye.com/blog/881972 在测试基于 DirectFB+Gstreamer 的视频联播系统的一个 Demo 的时候,其中大量使用 system 调用的语 ...
- Linux中的defunct进程(僵尸进程)
一.什么是defunct进程(僵尸进程)?在 Linux 系统中,一个进程结束了,但是他的父进程没有等待(调用wait / waitpid)他,那么他将变成一个僵尸进程.当用ps命令观察进程的执行状态 ...
- 杀死Linux中的defunct进程(僵尸进程)的方法指南
杀死Linux中的defunct进程(僵尸进程)的方法指南_LINUX_操作系统_脚本之家https://www.jb51.net/LINUXjishu/457748.html 这样能看到僵尸进程.
- Kill杀死Linux中的defunct进程(僵尸进程)
一.什么是defunct进程(僵尸进程)? 在 Linux 系统中,一个进程结束了,但是他的父进程没有等待(调用wait / waitpid)他,那么他将变成一个僵尸进程.当用ps命令观察进程的执行状 ...
- JAVA程序CPU 100%问题排查
做JAVA开发的同学一定遇到过的爆表问题,看这里解决 https://www.cnblogs.com/qcloud1001/p/9773947.html 本文由净地发表于云+社区专栏 记一次Ja ...
- (转)[sh] <defunct>进程
我也碰到僵尸进程了,必须通过关闭僵尸进程的父进程或者重启解决. 来自:http://jerrybear.blog.51cto.com/629421/387846 今日发现服务器上多了一些名称为[sh] ...
- 如何杀死defunct进程
原文: How to kill defunct process 译者: youngsterxyf defunct进程是指出错损坏的进程,父子进程之间不会再通信.有时,它们会演变成“僵尸进程”,存留在你 ...
随机推荐
- JDK1.8的HashMap数据结构及红黑树
在JDK1.6,1.7中,HashMap的实现都是用基础的“拉链法”去实现,即数组+链表的形式.如下图:通过不同的hash值,来对数据进行分配存储. 关于HashMap的Entry长度,可以参考htt ...
- C#中的根据实体增删改操作
在日常操作中,我们经常会对一些数据进行批量更新, 我在使用EF的时候,没有找到比较好的批量更新的解决方案, 便参考了张占岭前辈的博客,整合了这么一个简略版的使用实体类生成数据库增删改SQL的操作类 在 ...
- 配置phpmyadmin连接远程 MySQL数据库
引言:1.phpmyadmin程序所在服务器:192.168.1.1,访问地址为:http://192.168.1.1/phpmyadmin2.MySQL数据库所在服务器:192.168.1.2, ...
- Java第二次作业第三题
四叶玫瑰线的图形设计:当用鼠标拖拽改变窗口大小时,四叶玫瑰线会重新绘制 package naizi; import java.awt.*; import java.awt.event.*; impor ...
- spring集成mybatis-plus
一.mybatis-plus 使用mybatis-plus可以轻松实现通用crue.通用service,不用再在xml.dao.service里写增删改查的代码(需要写特殊方法的时候可以按原先的来), ...
- 制作mysql大数据表验证覆盖索引
昨天跟同事聊起数据表性能的问题,能不能仅用覆盖索引实现数据的汇总统计.找了一个开发环境已有的数据表进行测试,通过explain命令,能看到mysql通过覆盖索引就能实现sum的需求,而无须去读取实际行 ...
- 【Jsp】利用iframe实现action不跳转
<form role="form" target="id_frame" action="dk" method="post&q ...
- logback.xml配置文件解析一
配置文件主要结构如下: <?xml version="1.0" encoding="utf-8"?> <configuration> & ...
- Hive之行转列与列转行
行转列 原始数据: 需求: 把星座和血型一样的人归类到一起.结果如下: 射手座,A 大海|凤姐 白羊座,A 孙悟空|猪八戒 白羊座,B 宋宋 实现: vi person_info.txt 孙悟空 白羊 ...
- 远程桌面连接mstsc 超扎心
搞了一天问了几千个人.最后终于有一位大佬解决了.扎心啊. http://jingyan.baidu.com/article/39810a23edc48bb637fda672.html 就是服务没打开.