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

如何优雅的杀掉一个进程

廖万里2年前 (2022-10-27)文章74152

前言

在我们通常使用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 转载需授权!

分享到:

添加博主微信共同交流探讨信息差网赚项目: 19528888767 , 请猛戳这里→点我添加

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

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

同样是系统为什么有的叫OS有的叫UI

同样是系统为什么有的叫OS有的叫UI

大家有没有注意到现在手机系统虽然主要分iOS和安卓两大阵营,但如果从叫法的后缀上来区分的话其实是分为OS和UI。那么,你有没有想过为什么有些手机的系统后缀是OS?有些系统的后缀是UI呢?他们之间的区别在于哪些呢?同样是系统为什么有的叫OS有的叫UI1、字面意思不同OS的英文全称为Operation...

如何把备用手机号,改为移动便宜的8元套餐,不用去营业厅

如何把备用手机号,改为移动便宜的8元套餐,不用去营业厅

手机可以说是我们最常用的一个通讯工具,比如说我们这几年,可能会用过多部手机,也有多个手机号,但每个朋友可能都会有一台或者两台的备用机,里面只是保号使用,今天给大家分享技巧是如何把我们的备用手机号,设置为一个最低的一个保号套餐,这样的话就会少花冤枉钱,相信这个技巧呢,会对大家有很大的帮助,大家可以点赞...

每晚泡脚15分钟,五年下来会有哪些变化?

每晚泡脚15分钟,五年下来会有哪些变化?

剑心学者10月07日关注我是每到秋季与冬季才开始泡脚,尽量做到每晚都泡。当然经期除外。每次用艾叶、生姜、花椒煮沸后加水泡15到20分钟左右。泡到全身发热为止。坚持两年了。我身体湿气较重,随时舌头都有齿痕,夏天怕热冬天怕冷,手脚冰凉的。现在最大改变就是晚上睡觉不穿袜子,脚也不冰凉。还是有一定的效果。当...

如何让自己的努力更有效率?

如何让自己的努力更有效率?

收到了某个朋友发来的困惑咨询,抽象出来后整理出如下问题:为何自己很努力但觉得没有成长,做了很多事情却感觉没有核心竞争力,有浑身的精力不知道该往何处发力,应该如何破局?我是一名技术型产品经理,已经工作了3年,但是感觉自己陷入了成长迷茫期。 团队很重视技术,我花了很多时间来弥补技术知识,但是发现干不过研...

六年前端面试报告

六年前端面试报告

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

小米12SPro深度体验评测,什么叫“水 桶 旗 舰”啊?

小米12SPro深度体验评测,什么叫“水 桶 旗 舰”啊?

首先,依然是祖传一句话评价:补齐了唯一短板的水桶旗舰。其实就像年初的时候评价小米12Pro一样,今年的12系列看得出小米是在努力的优化体验,而不是单纯的堆叠参数,所以更注重手感的12和12S,以及徕卡加持下算法提升、影调有明显改善的小米12S Ultra便应运而生,至于今年的12Pro和12S Pr...

发表评论

访客

看不清,换一张

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