更加优雅的“ ps aux | grep -v grep”


177

当我检查流程列表并“ grep”列出那些对我来说很有趣的流程时,其grep本身也会包含在结果中。例如,列出终端:

$ ps aux  | grep terminal
user  2064  0.0  0.6 181452 26460 ?        Sl   Feb13   5:41 gnome-terminal --working-directory=..
user  2979  0.0  0.0   4192   796 pts/3    S+   11:07   0:00 grep --color=auto terminal

通常我ps aux | grep something | grep -v grep用来摆脱最后一个条目...但是它并不优雅 :)

您是否有更优雅的技巧来解决此问题(将所有命令包装到单独的脚本中,这也不错)


4
对于它的价值,这是一个古老的常见问题解答。请参阅faqs.org/faqs/unix-faq/faq/part3中的
13年

1
感谢您的参考。那是他们的方法:ps ux | awk '/name/ && !/awk/ {print $2}'
Jakub M.

grep -v grep做什么?
Jwan622 '17

2
@ Jwan622 从grep结果中grep -v grep排除grep。如果将grep与ps结合使用,则也会显示grep进程(带有grep参数),使结果混乱。grep -v grep是避免这种情况的一种常用方法
Eugene Platonov

Answers:


282

通常的技术是这样的:

ps aux | egrep '[t]erminal'

这将匹配包含的行terminal,但egrep '[t]erminal'不匹配!它还可以在多种 Unix版本上使用。


1
这对我适用于任意字符串,但不适用于用户名,例如ps aux | grep '[r]oot'。有人知道为什么吗?
kxsong 2014年

2
@kxsong:| grep '[t]erminal'选择包含单词“ terminal”的任何行,而不将单词“ terminal”放入过程列表。您想达到什么目的,| grep '[r]oot'以及它如何不起作用?可能会有更好的解决方案。
Johnsyweb 2014年

3
如果我错了,请纠正我,但这也适用于grepped字符的任何位置:ps aux | grep“ te [r]
minal

2
出色的技巧(我想我应该这样称呼,因为aux / grep作者可能没有想到这种情况。)
Michael Trouw

2
@JamieJag:好吧...就像我在帖子中说的那样,grep '[t]erminal'将匹配包含的行terminal。来自的输出ps aux将带有一行grep '[t]erminal'(带有方括号),该行不包含字符串terminal(没有相同的字符串)。
Johnsyweb '16

57

使用pgrep。更可靠。


pgrep如果我看上去例如,对于惯于工作ps aux | grep 'ssh options'
的Jakub M.

12
Jakub M .:那pgrep -f呢?
丘卢

4
@ jakub-m默认情况下pgrep仅将模式与进程名称匹配。要与整个命令匹配,请使用该-f标志。
bluecollarcoder 2014年

3
pgrep仅返回进程ID。
Melab 2015年

1
使用了旧版本pgrep -fl(但如果不-f匹配完整的cmdline,则无法看到完整的cmdline,详细信息:serverfault.com/a/619788/44183)。但是可以,如果您需要除pid,cmdline之外的任何其他信息,则需要ps。可以将它们组合:ps -p $(pgrep -f foo)
贝尼·切尔尼亚夫斯基-帕斯金

29

该答案基于先前的pgrep 答案。它还建立在另一个答案结合使用的pspgrep。以下是一些相关的培训示例:

$ pgrep -lf sshd
1902 sshd

$ pgrep -f sshd
1902

$ ps up $(pgrep -f sshd)
USER       PID %CPU %MEM    VSZ   RSS TTY      STAT START   TIME COMMAND
root      1902  0.0  0.1  82560  3580 ?        Ss   Oct20   0:00 /usr/sbin/sshd -D

$ ps up $(pgrep -f sshddd)
error: list of process IDs must follow p
[stderr output truncated]

$ ps up $(pgrep -f sshddd) 2>&-
[no output]

以上可以用作功能

$ psgrep() { ps up $(pgrep -f $@) 2>&-; }

$ psgrep sshd
USER       PID %CPU %MEM    VSZ   RSS TTY      STAT START   TIME COMMAND
root      1902  0.0  0.1  82560  3580 ?        Ss   Oct20   0:00 /usr/sbin/sshd -D

用比较psgrep。不会打印出有用的标题行:

$  ps aux | grep [s]shd
root      1902  0.0  0.1  82560  3580 ?        Ss   Oct20   0:00 /usr/sbin/sshd -D

参见github.com/blueyed/oh-my-zsh/blob/…(但特定于Zsh)。
2015年

@blueyed,我已经用bash函数定义更新了答案。
Acumenus

另请注意-d指定分隔符的选项;例如ps -fp$(pgrep -d , getty)
Toby Speight

1
我使用此方法:ps uxp `pgrep <process>` 请注意,p该参数必须是最后一个参数(即pux不会起作用)
KFL

9

您可以在ps命令中进行过滤,例如

ps u -C gnome-terminal

(或通过/ proc和find等进行搜索)


1
请注意,这适用于GNU的ps(Linux),但不适用于BSD ps。
2013年

即使使用GNU ps,这也是不正确的。 ps -C <command>将匹配确切的命令。当与使用ax选择将报告的所有进程,因为ax清单过程中除了该组通过其他方式匹配的进程。
Animism 2013年

@Animism是的。谢谢,我修好了。
安德里亚斯·弗里斯

仅当确切指定了进程名称时,此方法才有效。对于部分匹配(例如logindfor systemd-logind)或参数匹配不起作用。
Acumenus '16

4

一种选择

ps -fC terminal

这里的选项:

 -f        does full-format listing. This option can be combined
           with many other UNIX-style options to add additional
           columns. It also causes the command arguments to be
           printed. When used with -L, the NLWP (number of
           threads) and LWP (thread ID) columns will be added. See
           the c option, the format keyword args, and the format
           keyword comm.

 -C cmdlist     Select by command name.
                This selects the processes whose executable name is
                given in cmdlist.

6
一个较小的缺点(与OP的问题并没有真正的关系)是,它不会向您显示Tomcat之类的内容,而Tomcat实际上是作为Java运行的,带有一系列参数。
查尔斯·伍德

-C@Andreas Frishe在一年半前发布的答案中已经建议使用该选项……
Piotr Dobrogost

3

使用方括号将字符括在搜索模式中会排除该grep过程,因为该字符不包含匹配的正则表达式。

$ ps ax | grep 'syslogd'
   16   ??  Ss     0:09.43 /usr/sbin/syslogd
18108 s001  S+     0:00.00 grep syslogd

$ ps ax | grep '[s]yslogd'
   16   ??  Ss     0:09.43 /usr/sbin/syslogd

$ ps ax | grep '[s]yslogd|grep'
   16   ??  Ss     0:09.43 /usr/sbin/syslogd
18144 s001  S+     0:00.00 grep [s]yslogd|grep

3

免责声明:我是这个工具的作者,但是...

我会用px

~ $ px atom
  PID COMMAND          USERNAME   CPU RAM COMMANDLINE
14321 crashpad_handler walles   0.01s  0% /Users/walles/Downloads/Atom.app/Contents/Frameworks/Electron Framework.framework/Resources/crashpad_handler --database=
16575 crashpad_handler walles   0.01s  0% /Users/walles/Downloads/Atom.app/Contents/Frameworks/Electron Framework.framework/Resources/crashpad_handler --database=
16573 Atom Helper      walles    0.5s  0% /Users/walles/Downloads/Atom.app/Contents/Frameworks/Atom Helper.app/Contents/MacOS/Atom Helper --type=gpu-process --cha
16569 Atom             walles   2.84s  1% /Users/walles/Downloads/Atom.app/Contents/MacOS/Atom --executed-from=/Users/walles/src/goworkspace/src/github.com/github
16591 Atom Helper      walles   7.96s  2% /Users/walles/Downloads/Atom.app/Contents/Frameworks/Atom Helper.app/Contents/MacOS/Atom Helper --type=renderer --no-san

除了使用明智的命令行界面查找进程外,它还执行许多其他有用的操作,在项目页面上提供更多详细信息。

适用于Linux和OS X,易于安装:

curl -Ls https://github.com/walles/px/raw/python/install.sh | bash

0

根据最终用例,您通常希望改用Awk。

ps aux | awk '/[t]erminal/'

当您遇到类似问题时,尤其如此

ps aux | grep '[t]erminal' | awk '{print $1}'  # useless use of grep!

显然,正则表达式可以轻松地纳入Awk脚本中:

ps aux | awk '/[t]erminal/ { print $1 }'

但实际上,不要自己重塑。pgrep和朋友在一起已经有很长一段时间了,与大多数临时重新实现相比,处理整个问题空间要好得多。


-2

另一个选择是编辑您的.bash_profile(或其他保留bash别名的文件)以创建一个从结果中剔除'grep'的函数。

function mygrep {
grep -v grep | grep --color=auto $1
}

alias grep='mygrep'

grep -v grep必须是第一,否则你--color=auto不会出于某种原因。

如果您正在使用bash,则此方法有效;如果您使用的是其他外壳YMMV。


1
别名函数是什么?只是做function grep { command grep -v grep | command grep --color=auto "$@"; }(也请注意参数和引号的修复)。但是,这样做的缺点是,任何非ps调用都grep将不再起作用(参数传递不正确)。无论如何,一个更有用的功能将是修改正则表达式以使其自身不匹配,而不是单独grepgrep结果中过滤出来的功能。当然,为几十年前已充分解决的问题发明新的解决方案并不是很有效。
2015年
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.