如何在unix / linux系统上将命令行args传递给正在运行的进程?
题
在SunOS上有pargs
命令,用于打印传递给正在运行的进程的命令行参数。
其他Unix环境中是否有类似的命令?
解决方案
有几种选择:
ps -fp <pid>
cat /proc/<pid>/cmdline | sed -e "s/\x00/ /g"; echo
Linux上有/proc/<pid>
的更多信息,请看一下。
在其他Unix上,事情可能会有所不同。 ps
命令可以在任何地方使用,/proc
东西是特定于操作系统的。例如,在AIX上,cmdline
中没有<=>。
其他提示
这样可以解决问题:
xargs -0 < /proc/<pid>/cmdline
如果没有xargs,参数之间将没有空格,因为它们已被转换为NUL。
完整命令行
对于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
在 Linux
上cat /proc/<pid>/cmdline
获取进程的命令行(包括args),但所有空格都更改为NUL字符。
在Linux中使用空格打印/proc/PID/cmdline
的另一种变体是:
cat -v /proc/PID/cmdline | sed 's/\^@/\ /g' && echo
以这种方式cat
将 NULL字符打印为^@
,然后将其替换为使用空间sed
; echo
打印换行符。
不要使用多个命令来编辑流,只需使用一个 - tr将一个字符翻译成另一个字符:
tr '\0' ' ' </proc/<pid>/cmdline
除了上述所有转换文本的方法之外,如果只使用“字符串”,默认情况下它会在单独的行上输出。有了额外的好处,它还可以防止任何可能扰乱终端出现的字符。
两个输出都在一个命令中:
strings / proc // cmdline / proc // environ
真正的问题是......有没有办法在Linux中查看已经更改的进程的真实命令行,以便cmdline包含已更改的文本而不是运行的实际命令。
在Solaris上
ps -eo pid,comm
类似可以在类似unix的系统上使用。
在Linux上,使用bash,输出为引用的args,这样你就可以编辑命令并重新运行它
</proc/"${pid}"/cmdline xargs --no-run-if-empty -0 -n1 \
bash -c 'printf "%q " "${1}"' /dev/null; echo
在Solaris上,使用bash(使用3.2.51(1)-exlease测试)并且没有gnu userland:
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
}
输出:
<*>您可以简单地使用:
ps -o args= -f -p ProcessPid
尝试<!>“; ps -n <!>”;在linux终端中。这将显示:
1.所有进程运行,命令行及其PID
- 程序启动流程。 醇>
之后您将知道要杀死哪个进程
如果你想尽可能长(不确定有什么限制),类似于Solaris' pargs ,您可以在Linux <!>放大器上使用它; OSX:
ps -ww -o pid,command [-p <pid> ... ]