如何在Unix/Linux系统中获取传递给运行进程的命令行参数?

238

在SunOS上,有一个名为pargs命令,它会打印运行进程所传递的命令行参数。

其他Unix环境中有类似的命令吗?


7
该命令用于将进程号为< pid >的进程的命令行参数中的所有null字符替换为空格,并输出到标准输出流。 - Dmitry Grigoryev
14个回答

361

有几个选项:

ps -fp <pid>
cat /proc/<pid>/cmdline | sed -e "s/\x00/ /g"; echo

在Linux中,/proc/<pid>中有更多信息,请查看。

在其他Unix上可能会有所不同。 ps命令可以在任何地方使用,但/proc是特定于操作系统的。例如,在AIX上,/proc中没有cmdline


62
在Linux上,你可能需要使用-ww选项(例如 ps -ww -fp <pid>)来指定宽输出,因为如果有多个命令,它们可能会被截断。 - Silfheed
4
-ww 选项允许访问完整的命令行参数(只要内核存储了)。参见:如何获取Solaris和BSD进程的未截断命令行参数ps选项 - GuruM
3
cat /proc/<pid>/cmdline 在 Cygwin 上也适用,因为在 ps 的任何选项中都不显示命令行参数。 - lechup
3
在Linux上,如果你只需要获取args,命令是ps -o args -p <pid>,它只会打印出args,或者如果你只需要查看cmd,可以使用选项-o cmd。尝试读取/proc/<pid>/cmdline对于非特权用户并不总是有效的。ps工具则可以工作。 - alvits
3
提示:/proc/<pid>/cmdline 的长度是有限制的(硬编码为PAGE_SIZE内核参数的值),因此更长的命令行仍然显示截断!有关详细信息,请参见https://dev59.com/_3VC5IYBdhLWcg3wvTxa。您可以使用`getconf PAGE_SIZE`查询您的内核设置,通常为4096。 - t0r0X
显示剩余4条评论

73

这样做就可以了:

xargs -0 < /proc/<pid>/cmdline

如果没有使用xargs,参数之间将没有空格,因为它们已被转换为NUL字符。


它仍然截断我的输出。有什么建议吗? - johnsam
从未注意到任何截断 - 你能举个例子吗? - Michael Böckling
这样,你无法确定它是内联空格还是参数边界。 - ivan_pozdeev

28

完整的命令行

对于Linux和Unix系统,您可以使用ps -ef | grep process_name获取完整的命令行。

在SunOS系统上,如果您想要获取完整的命令行,则可以使用

/usr/ucb/ps -auxww | grep -i process_name

要获取完整的命令行,您需要成为超级用户。

参数列表

pargs -a PROCESS_ID

将详细列出传递给进程的参数列表。它将以以下方式输出参数数组:

argv[o]: first argument
argv[1]: second..
argv[*]: and so on..

我没有在Linux中找到任何类似的命令,但是我会使用以下命令来获得类似的输出:

tr '\0' '\n' < /proc/<pid>/environ

我正在寻找的命令是 ps -ef - Timofey

25

Linux

cat /proc/<pid>/cmdline

该函数输出进程 <pid> 的命令行(包括参数),每个记录以一个 NUL 字符作为结束符。

一个 Bash Shell 示例

$ mapfile -d '' args < /proc/$$/cmdline
$ echo "#${#args[@]}:" "${args[@]}"
#1: /bin/bash
$ echo $BASH_VERSION
5.0.17(1)-release

3
空白字符不被删除,而是被替换为 NUL 字符。 - bdonlan
@bdonlan 啊,我没有检查过。好发现! - lothar
5
使用命令"xargs -0 echo < /proc/<pid>/cmdline"可显示进程< pid >的命令行参数。使用/proc/<pid>/environ也可以实现这一功能,不过建议在该命令中添加"-n 1"参数。 - camh
我的系统上没有/proc文件系统 :( 还有其他解决方案吗? - Hemant
我的是一个带有很长很长参数的Java进程。它仍然截断我的输出。有什么建议吗? - johnsam
显示剩余2条评论

19
你可以使用pgrep-f(完整命令行)以及-l(长描述):
pgrep -l -f PatternOfProcess

这种方法与其他任何响应的关键区别在于:它适用于CygWin,因此您可以使用它来获取在Windows下运行的任何进程的完整命令行(如果要获取有关任何提升/管理员进程的数据,请作为elevated执行)。在Windows上执行此操作的任何其他方法都更加麻烦( 例如 )。
此外,在我的测试中,pgrep方式是唯一有效的系统,可获取在CygWin的python中运行的脚本的完整路径。


这个命令实际上会打印出原始可执行文件的名称:`$ exec -a fakename bash & [1] 14102[1]+ Stopped exec -a fakename bash $ xargs -0 < /proc/14102/cmdline; fakename $ pgrep -l -f fakename; 14102 bash` - unhammer
使用 pgrep from procps-ng 3.3.153.3.12 对我不起作用。只打印pid和程序名称,没有参数。 - Socowi

4
在Linux中,打印带有空格的/proc/PID/cmdline的另一种变体是:
cat -v /proc/PID/cmdline | sed 's/\^@/\ /g' && echo

通过这种方式,cat^@ 的形式打印 NULL 字符,然后你可以使用 sed 将其替换为一个空格;echo 打印一个新行。


你也可以使用 **cat -v /proc/PID/cmdline | sed 's/^@/\n/g'**。这将用换行符替换空字符。这样,每个参数都会被打印到自己的一行上。这样做可以更容易地区分一个参数和另一个参数。 - TSJNachos117

4

与其使用多个命令来编辑流,不如只使用一个 - tr将一个字符转换为另一个字符:

tr '\0' ' ' </proc/<pid>/cmdline

3

ps -eo pid,args 命令可以打印进程的PID和完整命令行。


2
您可以简单地使用以下代码:
ps -o args= -f -p ProcessPid

1
在Linux上,使用bash,将输出作为带引号的参数,以便您可以编辑命令并重新运行它。
</proc/"${pid}"/cmdline xargs --no-run-if-empty -0 -n1 \
    bash -c 'printf "%q " "${1}"' /dev/null; echo

在Solaris上,使用bash(已测试版本为3.2.51(1)-release),并且没有gnu用户空间:
IFS=$'\002' tmpargs=( $( pargs "${pid}" \
    | /usr/bin/sed -n 's/^argv\[[0-9]\{1,\}\]: //gp' \
    | tr '\n' '\002' ) )
for tmparg in "${tmpargs[@]}"; do
    printf "%q " "$( echo -e "${tmparg}" )"
done; echo

Linux bash示例(粘贴到终端中):

{
## setup intial args
argv=( /bin/bash -c '{ /usr/bin/sleep 10; echo; }' /dev/null 'BEGIN {system("sleep 2")}' "this is" \
    "some" "args "$'\n'" that" $'\000' $'\002' "need" "quot"$'\t'"ing" )

## run in background
"${argv[@]}" &

## recover into eval string that assigns it to argv_recovered
eval_me=$(
    printf "argv_recovered=( "
    </proc/"${!}"/cmdline xargs --no-run-if-empty -0 -n1 \
        bash -c 'printf "%q " "${1}"' /dev/null
    printf " )\n"
)

## do eval
eval "${eval_me}"

## verify match
if [ "$( declare -p argv )" == "$( declare -p argv_recovered | sed 's/argv_recovered/argv/' )" ];
then
    echo MATCH
else
    echo NO MATCH
fi
}

输出:

MATCH

Solaris Bash 示例:
{
## setup intial args
argv=( /bin/bash -c '{ /usr/bin/sleep 10; echo; }' /dev/null 'BEGIN {system("sleep 2")}' "this is" \
    "some" "args "$'\n'" that" $'\000' $'\002' "need" "quot"$'\t'"ing" )

## run in background
"${argv[@]}" &
pargs "${!}"
ps -fp "${!}"

declare -p tmpargs
eval_me=$(
    printf "argv_recovered=( "
    IFS=$'\002' tmpargs=( $( pargs "${!}" \
        | /usr/bin/sed -n 's/^argv\[[0-9]\{1,\}\]: //gp' \
        | tr '\n' '\002' ) )
    for tmparg in "${tmpargs[@]}"; do
        printf "%q " "$( echo -e "${tmparg}" )"
    done; echo
    printf " )\n"
)

## do eval
eval "${eval_me}"


## verify match
if [ "$( declare -p argv )" == "$( declare -p argv_recovered | sed 's/argv_recovered/argv/' )" ];
then
    echo MATCH
else
    echo NO MATCH
fi
}

输出:

MATCH

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接