[英]Why doesn't “xargs sudo perf top” work as expected?
我想分析一个进程,所以首先获取它的 pid,然后使用“ perf top
”来检查它:
$ pgrep program
14472
$ sudo perf top -p 14472
它按预期工作:
然后我想使用 pipe 来连接这两个命令,所以我使用xargs
:
$ pgrep program | sudo xargs perf top -p
但这一次“ perf top
”似乎无法正常工作: 我比较了这两个操作的过程:
(1)分别运行pgrep
和perf
:
$ ps -ef | grep perf
root 18468 16827 0 09:34 pts/3 00:00:00 sudo perf top -p 14472
root 18469 18468 91 09:34 pts/3 00:00:06 perf top -p 14472
nanxiao 18477 18295 0 09:34 pts/4 00:00:00 grep --color=auto perf
(2) 使用xargs
连接pgrep
和perf
:
$ ps -ef | grep perf
nanxiao 18250 16827 0 09:32 pts/3 00:00:00 xargs sudo perf top -p
root 18251 18250 0 09:32 pts/3 00:00:00 sudo perf top -p 14472
root 18252 18251 87 09:32 pts/3 00:01:47 perf top -p 14472
nanxiao 18442 18295 0 09:34 pts/4 00:00:00 grep --color=auto perf
恕我直言,看起来一样。 任何人都可以提供一些线索吗? 提前致谢!
PS,我的操作系统是CentOS 7
。
再次检查手册后,我发现-o
选项可以解决此问题:
-o, --open-tty 在执行命令之前,在子进程中重新打开标准输入为 /dev/tty。 如果您希望 xargs 运行交互式应用程序,这将很有用。
命令是这样的:
$ pgrep program | sudo xargs -o perf top -p
但不幸的是, CentOS 7
的xargs
有点老了,没有提供这个选项。
根本原因是:没有-o
选项perf
stdin
的标准输入是/dev/null
:
$ sudo lsof -p 1495
......
perf 1495 root 0r CHR 1,3 0t0 2052 /dev/null
......
并且性能在perf
()中被阻止:
......
FD_ZERO(&read_set);
FD_SET(0, &read_set);
if (delay_secs) {
timeout.tv_sec = delay_secs;
timeout.tv_usec = 0;
}
err = select(1, &read_set, NULL, NULL, ptimeout);
if (err == 0)
return K_TIMER;
if (err == -1) {
if (errno == EINTR)
return K_RESIZE;
return K_ERROR;
}
key = SLang_getkey();
if (key != K_ESC)
return key;
......
读取/dev/null
将返回EOF
,然后select()
将返回1
。
使用-o
选项, perf
程序的标准输入是/dev/tty
stdin
$ sudo lsof -p 1394
......
perf 1394 root 0u CHR 136,25 0t0 28 /dev/pts/25
......
在上面的代码中, select()
将返回0
,整个 function 将相应地返回。
更好的方法是直接在pgrep
的 output 上运行top
,而不是通过xargs
管道。 我相信默认情况下top
命令不会通过标准输入读取信息
sudo perf top -p "$(pgrep program)"
这样$(..)
返回pgrep
命令的 output 并且返回的值作为位置参数值传递给-p
标志。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.