我经常需要在编程期间终止一个进程。

我现在的做法是:

[~]$ ps aux | grep 'python csp_build.py'
user    5124  1.0  0.3 214588 13852 pts/4    Sl+  11:19   0:00 python csp_build.py
user    5373  0.0  0.0   8096   960 pts/6    S+   11:20   0:00 grep python csp_build.py
[~]$ kill 5124

如何自动提取进程id并在同一行中杀死它?

是这样的:

[~]$ ps aux | grep 'python csp_build.py' | kill <regex that returns the pid>

当前回答

一个只使用awk(和ps)的方法:

ps aux | awk '$11" "$12 == "python csp_build.py" { system("kill " $2) }'

通过使用字符串相等性测试,我防止匹配这个过程本身。

其他回答

解决方案是用精确的模式过滤进程,解析pid,并构造一个参数列表来执行kill进程:

ps -ef  | grep -e <serviceNameA> -e <serviceNameB> -e <serviceNameC> |
awk '{print $2}' | xargs sudo kill -9

文件说明:

Ps实用程序显示标题行,后面的行包含 关于拥有控制终端的所有进程的信息。

-e显示其他用户的进程信息

-f显示uid、pid、父pid、最近CPU使用率、进程启动

grep实用程序搜索任何给定的输入文件,选择行 -e pattern,——regexp=pattern 指定搜索输入期间使用的模式:输入 如果匹配任何指定的模式,则选择该行。 当使用多个-e选项时,此选项最有用 指定多个模式,或者当一个模式以破折号开始时 (“-”)。

Xargs -构造参数列表并执行实用程序

终止—终止进程或给进程发信号

9号信号- KILL(不可捕捉,不可忽略的杀死)

例子:

ps -ef  | grep -e node -e loggerUploadService.sh -e applicationService.js |
awk '{print $2}' | xargs sudo kill -9

我的任务是杀死所有与regexp匹配的放在特定目录中的内容(在selenium测试之后,并不是所有内容都停止了)。这招对我很管用:

for i in `ps aux | egrep "firefox|chrome|selenium|opera"|grep "/home/dir1/dir2"|awk '{print $2}'|uniq`; do kill $i; done

根据https://stackoverflow.com/a/3510879/15603477的回答。小的优化。

ps aux | grep 'python csp_build.py' | head -1 | tr -s ' ' | cut -d " " -f 2 | xargs kill

使用tr -s ' '将多个空白(如果有的话)压缩为一个空白。

如果你遇到不允许的行动,请跟着>> https://unix.stackexchange.com/questions/89316/how-to-kill-a-process-that-says-operation-not-permitted-when-attempted

在bash中,只使用问题(1)中列出的基本工具,你应该能够做到:

kill $(ps aux | grep '[p]ython csp_build.py' | awk '{print $2}')

其工作详情如下:

ps给出了所有进程的列表。 基于搜索字符串的grep过滤器[p]是一个阻止您获取实际grep进程本身的技巧。 awk只给你每行的第二个字段,也就是PID。 $(x)构造意味着执行x,然后获取其输出并将其放在命令行上。在上面的构造中,ps管道的输出是进程id列表,因此您最终会得到kill 1234 1122 7654这样的命令。

以下是它实际运行的文字记录:

pax> sleep 3600 &
[1] 2225
pax> sleep 3600 &
[2] 2226
pax> sleep 3600 &
[3] 2227
pax> sleep 3600 &
[4] 2228
pax> sleep 3600 &
[5] 2229
pax> kill $(ps aux | grep '[s]leep' | awk '{print $2}')
[5]+  Terminated              sleep 3600
[1]   Terminated              sleep 3600
[2]   Terminated              sleep 3600
[3]-  Terminated              sleep 3600
[4]+  Terminated              sleep 3600

你可以看到它终止了所有的休眠。

更详细地解释一下grep '[p]ython csp_build.py'位:当您执行sleep 3600并且后面跟着ps -ef | grep sleep时,您往往会得到两个包含sleep的进程,sleep 3600和grep sleep(因为它们都包含sleep,这不是火箭科学)。

然而,ps -ef | grep '[s]leep'不会创建一个包含sleep的grep进程,而是使用命令grep '[s]leep'创建一个grep进程,这里有一个棘手的地方:grep不会找到这个进程,因为它正在寻找正则表达式“字符类[s](基本上就是s)中的任何字符后跟leep”。

换句话说,它在寻找sleep但是grep进程是grep '[s]leep'里面没有sleep文本。

当我看到这个(有人在这里的SO),我立即开始使用它,因为

它比添加| grep -v grep少了一个进程;而且 这是优雅而狡猾的,一个罕见的组合:-)


(1)如果你不局限于使用这些基本工具,有一个漂亮的pgrep命令,它可以根据特定的标准查找进程(当然,假设你的系统上有这个命令)。

例如,您可以使用pgrep sleep来输出所有sleep命令的进程id(默认情况下,它与进程名匹配)。如果你想匹配ps中显示的整个命令行,你可以执行类似pgrep -f 'sleep 9999'这样的操作。

顺便说一句,如果执行pgrep,它不会列出自己,因此在这种情况下不需要上面所示的棘手的筛选方法。

通过使用-a显示完整的进程名,可以检查这些进程是否是您感兴趣的进程。您还可以使用-u或-u将范围限制为您自己的进程(或特定的用户集)。有关更多选项,请参阅pgrep/pkill的手册页。

一旦你满意了,它将只显示你感兴趣的进程,然后你可以使用pkill和相同的参数向所有这些进程发送信号。

从一个普通的PPID开始杀死我们自己的进程是相当频繁的,pkill与-P标志相关联对我来说是一个赢家。以@ghostdog74为例:

# sleep 30 &                                                                                                      
[1] 68849
# sleep 30 &
[2] 68879
# sleep 30 &
[3] 68897
# sleep 30 &
[4] 68900
# pkill -P $$                                                                                                         
[1]   Terminated              sleep 30
[2]   Terminated              sleep 30
[3]-  Terminated              sleep 30
[4]+  Terminated              sleep 30