当前位置:首页 > 文章 > 正文内容

如何优雅的杀掉一个进程

廖万里3年前 (2022-10-27)文章74238

前言

在我们通常使用linux操作系统的时候,经常会有这样的需求——杀死一个进程,比如说你一不小心启动了一个后台进程或者守护进程,而这个进程是你不需要的,因此你久想杀掉他,在本篇文章当中主要给大家介绍一些杀死进程的方法,以及这隐藏在这后面的原理。

你可以杀死哪些进程

在我们杀死一个进程的时候最好不要使用管理员权限,因为你可能会一不小心杀死系统当中一些很重要的进程。同时需要了解,在linux当中有很多与权限相关的操作,如果你只是一个普通的用户,那么你就只能杀死你自己的进程,不能够杀死别的用户的进程。但是root用户或者你有sudo权限,那么你就可以为所欲为了?。

杀死进程的基本原理:我们使用命令去杀死进程,本质上是通过一个进程(比如说kill命令的那个进程)给另外一个进程发送信号,当进程接收到信号的时候就会进行判断是哪个信号,然后根据不同的信号做出相应的行为。

信号前面表示代表不同信号的数值,比如说我们执行命令 kill -9 1234 就是将 9 这个值对应的信号 SIGKILL 发送给进程号等于 1234 的进程,在linux操作系统当中,常见的信号如下所示:


1) SIGHUP  2) SIGINT  3) SIGQUIT  4) SIGILL  5) SIGTRAP

6) SIGABRT  7) SIGBUS  8) SIGFPE  9) SIGKILL 10) SIGUSR1

11) SIGSEGV 12) SIGUSR2 13) SIGPIPE 14) SIGALRM 15) SIGTERM

16) SIGSTKFLT 17) SIGCHLD 18) SIGCONT 19) SIGSTOP 20) SIGTSTP

21) SIGTTIN 22) SIGTTOU 23) SIGURG 24) SIGXCPU 25) SIGXFSZ

26) SIGVTALRM 27) SIGPROF 28) SIGWINCH 29) SIGIO 30) SIGPWR

31) SIGSYS 34) SIGRTMIN 35) SIGRTMIN+1 36) SIGRTMIN+2 37) SIGRTMIN+3

38) SIGRTMIN+4 39) SIGRTMIN+5 40) SIGRTMIN+6 41) SIGRTMIN+7 42) SIGRTMIN+8

43) SIGRTMIN+9 44) SIGRTMIN+10 45) SIGRTMIN+11 46) SIGRTMIN+12 47) SIGRTMIN+13

48) SIGRTMIN+14 49) SIGRTMIN+15 50) SIGRTMAX-14 51) SIGRTMAX-13 52) SIGRTMAX-12

53) SIGRTMAX-11 54) SIGRTMAX-10 55) SIGRTMAX-9 56) SIGRTMAX-8 57) SIGRTMAX-7

58) SIGRTMAX-6 59) SIGRTMAX-5 60) SIGRTMAX-4 61) SIGRTMAX-3 62) SIGRTMAX-2

63) SIGRTMAX-1 64) SIGRTMAX

对于信号来说,进程可以有不同的应对行为,具体来说主要有以下三种:

  • 忽略这个信号。

  • 使用默认行为去处理这个信号,比如SIGINT和SIGTERM这两个信号的默认行为就是退出程序。

  • 自己定义函数捕获这个信号,我们可以自己写一个函数,并且使用系统调用将这个函数进行注册,当收到对应的信号的时候就去执行我们自己实现的函数,但是需要注意的是,并不是所有的信号我们都可以进行捕获的,比如说SIGKILL和SIGSTOP这两个信号。

程序的定位

我们通常可以使用 ps 和 top 两个命令进行程序的定位,在前面的两篇文章Linux命令系列之top——里面藏着很多鲜为人知的宝藏知识这才是使用ps命令的正确姿势当中我么已经仔细讨论过这个问题了!因此当我们想要杀死某个程序的时候我们可以通过上述两个命令进行程序的定位,找到我们想要杀死的进程的进程号和进程名字。

在本篇文章当中主要使用一个程序 group.c 作为例子,讲述各种 kill 命令的使用,他会 fork 出几个子进程,子进程的进程名和它的进程名都是 group.out,这个程序的源代码如下所示:




#include <stdio.h>

#include <unistd.h>



int main() {

for(int i = 0; i < 10; i++) {

if(!fork())

break;

}

printf("进程ID = %d 进程组ID = %d\n", getpid(), getpgid(0));

sleep(100);

return 0;

}

使用kill命令杀死进程

kill命令的使用方法如下所示:


kill [option] <pid> [...] # [option] 是参数选项比如 -9 pid 表示进程的进程id

发送一个 SIGINT 信号给进程 1234


kill -SIGINT 1234

或者

kill -2 1234

如果进程 1234 执行SIGINT的默认行为的话,那么进程1234就会退出,因为默认行为就是退出程序。

强制杀死进程 1234


kill -SIGKILL 1234

或者

kill -9 1234

因为信号 SIGKILL 是不能够被忽略或者捕获的,这个就是强制杀死程序,这条命令可以保证一定杀死进程,但是我们一般情况下最好不要使用这条命令,因为很多程序有他自己的逻辑,比如清理一些数据和系统资源,但是如果你不关心这些就无所谓了。

杀死所有你有权限杀死的进程


kill -9 -1

上面的命令当中 -1 的意思表示将 -9 这个信号发送给所有你有权限发送的进程,这个命令慎用。

有的时候我们会有一个需求就是杀死一个进程组里面的所有进程,我们可以使用命令


kill [option] -<pid> [...] # [option] 是参数选项比如 -9 pid 表示进程的进程id

就是在pid前面加上-表示杀死这个进程组,比如下面图中的示例:
如何优雅的杀掉一个进程

在 kill 命令当中如果没有指定发送那个信号的话,默认就发送SIGTERM信号,对应的数值等于15。

pkill和pgrep

pgrep 命令其实就是根据正则表达式列出相应的进程。默认他只会讲符合要求的进程的进程号列出来:


➜  pthreads pgrep out

3204266

3204268

3204269

3204270

3204271

3204272

3204273

比如说,列出所有含有 out 这个字符的进程:


➜  pthreads pgrep -l out

3204266 group.out

3204268 group.out

3204269 group.out

3204270 group.out

3204271 group.out

3204272 group.out

3204273 group.out

pgrep -l out

在上面的命令当中 out 表示一个字符串,我们也可以使用正则表达式,-l 表示列出进程执行的时候的命令。

在 pgrep 当中还有一个比较重要的选项 -u ,这个选项表示根据特定的用户筛选进程,比如说我们只要 abc 这个用户的进程,可以这样:


pgrep -u abc out # 也是选择含有 out 字符串的进程

pgrep 还有一个比较重要的选项就是 -v ,这个选项的意思就是除了符合正则表达式要求的其他进程,比如说我们想要将所有不含 out 的进程筛选出来,就可以使用如下命令:


pgrep -v out

如果我们想要统计一下符合要求的进程的个数我们可以使用下面的这个命令:


pgrep -c out # 统计一下含有 out 字符串的进程的个数

pkill 的使用方式和 pgrep 是一样的只不过 pgrep 是将进程的进程号列出来,而pkill是将一个发送给所有符合要求的进程,默认发送的信号是 SIGTERM 对应的值等于 15。
如何优雅的杀掉一个进程

比如在上图当中就发送一个默认信号SIGTERM给所有命令行当中还有字符串 out 的进程。因此 pkill 在批处理场景用的比较多,将还有某个特征的进程全部杀死。

如果你想指定具体发送那个信号格式和 kill 是一样的,比如发送 SIGKILL(-9)信号给含有字符串 out 的进程。


pkill -9 out

或者

pkill -SIGKILL out

使用killall命令

Killall 和 pkill 使用方法差不多,而且含义也一致,将符合条件的进程全部杀死。默认发送的信号也是SIGTERM,信号值等于15,但是pkill不同的是 killall 默认不开启正则表达式,我们需要通过 -r 选项启动正则表达式识别。

例子如下所示:
如何优雅的杀掉一个进程

如果我们想在不实用 -r 选项的情况下杀死进程,只能输入进程的全称了。
如何优雅的杀掉一个进程

在killall命令当中还有一些常用的选项:

选项含义
-u只杀死指定用户的进程,比如说 -u abc 只杀死用户 abc 满足要求的进程
-I字母大小写不敏感
-i交互模式,每次杀死进程都会询问是否杀死进程
-r表示使用正则表达式进行匹配
-数字或者信号名发送特定的信号,如下的例子所示


如何优雅的杀掉一个进程

为什么我们不能够捕获所有的信号

在前面的文章当中我们提到了,SIGKILL和SIGSTOP这两个信号是不能够被捕获的!试想一下,在你的系统的当中有一个病毒程序,他会不断的创建新的进程并且不断的申请系统资源,那么你还没有办法杀死他,你只能够眼睁睁的看着你的系统卡死。这种问题 linux 设计者早就想到了,基于这个问题就肯定需要有一种方式能够万无一失的杀掉进程,因为我们不能够让一个进程无休止的消耗系统资源而我们却无能为力,因此才出现了不能够被捕获的信号,因为这个能够确保杀死进程。

总结

在本篇文章当中主要给大家介绍各种 kill 命令的使用方法以及一些简单的信号含义的介绍,这部分内容在我们平常的学习工作过程当中经常会使用到,希望大家有所收获~


本文链接:https://www.kkkliao.cn/?id=154 转载需授权!

分享到:

版权声明:本文由廖万里的博客发布,如需转载请注明出处。


“如何优雅的杀掉一个进程” 的相关文章

外媒惊呼,继智能手机之后,中国在这一领域强势崛起!

外媒惊呼,继智能手机之后,中国在这一领域强势崛起!

作为一个80后,我清楚地记得,在功能手机的时代,中国的手机厂商一个能打的都没有,很多人为此努力过,但最后都折戟沉沙,黯淡收场。彼时的中国手机市场完全是三星、诺基亚、摩托罗拉等外资品牌的市场,仅存的中国手机厂商只能在夹缝中生存,靠生产杂牌手机苟延残喘。那时候我们多么希望能崛起一个自主品牌、民族品牌来收...

美设备大厂接连表态,反转出现了,外媒:华为将“打”出和平

美设备大厂接连表态,反转出现了,外媒:华为将“打”出和平

华为5G的横空出世,一举打破了美国数十年的通信霸权,也首次将中国通信带到了全球最顶端的位置,堪称是中国人的骄傲。然而,这却引起了老美的忌惮,它为了巩固自己的科技霸权地位,近些年频繁修改芯片规则,妄图通过断供来放倒华为。大家都知道,技术封锁是美帝霸凌主义惯用的手段,且屡试不爽,日本东芝和法国阿尔斯通的...

开始“反扑”了?中国院士正式宣布,厉害的不只有华为

开始“反扑”了?中国院士正式宣布,厉害的不只有华为

随着中国的这些年的不断发展,中国在各方都取得了很大的成就,现在的中国已经发生了翻天覆地的变化。在生活上。我们进入了电商时代,进入了移动支付时代;在科技上,我们进入了智能互联网时代,即将迎来的是全新的5G时代,还有很多很多中国制造都是值得我们骄傲的。特别值得提及的就是,现在中国制造已经成为了我们的一个...

六零后已经渐渐老了,都是独生子女家庭,以后怎样养老?

六零后已经渐渐老了,都是独生子女家庭,以后怎样养老?

我是68年的,今年54岁,我也只有一个女儿,以后怎么养老的问题,我在十年前就开始准备了,所以我的养老问题我现在一点都不担心,因为我早已经准备好了。我从上班到退休一直在银行储蓄柜工作。工作小柜台,人生大舞台,在几十年的工作中我见识了形形色色的人,也见过很多老人特别是没有退休金的老人晚年凄苦的生活。所以...

你最讨厌QQ什么?

你最讨厌QQ什么?

作为中国最早的社交软件之一, QQ承载着无数的青春。QQ最初诞生的时候,是为了方便人们之间的交流而诞生的。那时候 QQ作为我们联系的主要工具,人们之间可以进行即时通讯。我们每天都会在 QQ上和不同的人进行沟通互动。聊天的内容也十分的丰富,有的时候聊天的内容甚至超过了现在人们生活的内容。而其中最让人讨...

六年前端面试报告

六年前端面试报告

2022.10.20 在当前公司待了两年多,被离职了,拿了点赔偿金继续面试。薪资期望 13-15, 趁着今天1024整理下面试过程。上一次面试我是4年经验,简历也好改,加上两年经验,补上现公司项目就出去找工作了。简历改完后,首先分析下自己现阶段水平,大概能要多少,定一个期望薪资。再就是背面试题了。自...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。