在命令行上,我键入了一个命令,然后按Enter。它不输出任何东西。我如何知道它是否正在运行但尚未输出,或者正在要求用户输入?
cat
命令。只需cat
自行键入,它将等待来自stdin的输入,但不给出提示。许多其他命令的行为类似,因为它们期望从stdin或文件输入,但不区分不同的输入源(交互式终端,管道,文件...)。
在命令行上,我键入了一个命令,然后按Enter。它不输出任何东西。我如何知道它是否正在运行但尚未输出,或者正在要求用户输入?
cat
命令。只需cat
自行键入,它将等待来自stdin的输入,但不给出提示。许多其他命令的行为类似,因为它们期望从stdin或文件输入,但不区分不同的输入源(交互式终端,管道,文件...)。
Answers:
有几种方法:
尝试用信号通知输入结束:没有超级用户权限,很难知道引擎盖下发生了什么。可以按Ctrl+ 来完成d。规范模式下的终端和实用程序在read()
收到绑定到此组合键的EOT信号后,会将所有可用文本发送到syscall,并且如果没有输入- read()
返回负退出状态,大多数实用程序都将其接受为退出信号。因此,如果实用程序正在等待输入,它将在收到组合键时退出。否则,该实用程序要么正在运行任务,要么编写不正确。
监视系统调用:如果您具有超级用户特权,则可以strace
在另一个终端上运行以查看当前正在执行的操作。为此,您需要找出程序的PID。例如,在另一个终端选项卡中运行pgrep -f firefox
,例如,可能为1234,然后是sudo strace -f -p 1234
。如果您看到的输出停留在read()
syscall上,则意味着该命令可能正在等待输入。否则,如果看到syscall正在运行,则命令正在执行其他操作。有关用法的信息,请参阅相关问题,strace
以弄清长时间运行的命令是否已退出。
使用命令自己的方法:除其他外,诸如dd
使用信号之类的实用程序。例如,如果您使用kill -USR1 1234
(其中1234是正在运行的dd
命令的PID ),它将打印到stdout当前正在处理的字节数。当然,这首先需要了解命令的这种行为。上面的两种方法比较通用,不需要深入了解每个命令的行为(尽管最好总是知道您实际执行的是什么,否则您就有可能运行可能会造成损坏的命令)。
strace
方法:-)但是,更简单的方法也很有用(对于每个程序通用或特定)。其中一些没有超级用户权限就可以工作。示例:检查是否dd
正在做某事,并检查为什么grep --color asdf
静默等待。
dd
我将其添加。
它取决于程序以及如何调用它。
通常但并非总是会有提示,表明程序正在请求输入。
如果不确定,可以检查程序进程是否繁忙
使用CPU-使用top
或htop
读取或写入-使用 sudo iotop -o
程序完成后,您将看到外壳程序的提示。
running
我有一个shellscript,用于检查程序是否正在运行,现在我添加了一个选项,-s
以sudo strace -f -p <PID>
在找到...时使其运行(根据Sergiy Kolodyazhnyy的回答)。
shellscript使用
ps -ef
查找大多数程序systemctl is-active --quiet
找到一些程序并且如果您希望strace
在xterm
窗口中。
安装xterm
,如果你想使用strace
来观看节目的活动。
$ ./running
Usage: ./running <program-name>
./running <part of program name>
Examples: ./running firefox
./running term # part of program name
./running dbus
./running 'dbus-daemon --session' # words with quotes
./running -v term # verbose output
./running -s term # strace checks activity
如果您想方便地访问它,可以将shellscript安装running
到目录中PATH
。
#!/bin/bash
# date sign comment
# 2019-02-14 sudodus version 1.0
verbose=false
strace=false
if [ "$1" == "-v" ]
then
verbose=true
shift
fi
if [ "$1" == "-s" ]
then
strace=true
shift
fi
if [ $# -ne 1 ]
then
echo "Usage: $0 <program-name>
$0 <part of program name>
Examples: $0 firefox
$0 term # part of program name
$0 dbus
$0 'dbus-daemon --session' # words with quotes
$0 -v term # verbose output
$0 -s term # strace checks activity"
exit
fi
inversvid="\0033[7m"
resetvid="\0033[0m"
redback="\0033[1;37;41m"
greenback="\0033[1;37;42m"
blueback="\0033[1;37;44m"
runn=false
#tmpfil=$(mktemp)
tmpdir=$(mktemp -d)
tmpfil="$tmpdir/tmpfil"
vtfile="$tmpdir/vtfile"
vthead="$tmpdir/vthead"
# check by systemctl
systemctl is-active --quiet "$1"
if [ $? -eq 0 ]
then
echo "systemctl is-active:"
runn=true
fi
# check by ps
ps -ef | tr -s ' ' ' ' | cut -d ' ' -f 8- | grep "$1" | grep -vE -e "$0 *$1" -e "$0 *.* *$1" -e "grep $1" | sort -u > "$tmpfil"
#cat "$tmpfil"
if $verbose || $strace
then
ps -ef |head -n1 > "$vthead"
ps -ef | grep "$1" | grep -vE -e "$0 *.* *$1" -e "grep $1" | sort -u > "$vtfile"
fi
tmpstr=$(head -n1 "$tmpfil")
#echo "tmpstr=$tmpstr"
tmpess=$(grep -om1 "$1" "$tmpfil")
#echo "tmpess=$tmpess"
if [ "$tmpstr" == "$1" ] || [ "${tmpstr##*/}" == "$1" ] || [ "${1##*/}" == "${0##*/}" ] || [ "$tmpess" == "$1" ]
then
echo "ps -ef: active:"
runn=true
if $verbose
then
cat "$vthead" "$vtfile"
fi
elif test -s "$tmpfil"
then
if $runn
then
echo "----- consider also ------------------------------------------------------------"
if $verbose
then
cat "$vthead" "$vtfile"
else
cat "$tmpfil"
fi
echo "--------------------------------------------------------------------------------"
else
echo "----- try with: ----------------------------------------------------------------"
if $verbose
then
cat "$vthead" "$vtfile"
else
cat "$tmpfil"
fi
echo "--------------------------------------------------------------------------------"
fi
fi
if $runn
then
echo -en "$greenback '$1"
if [ "$tmpstr" != "$tmpess" ]
then
echo -n " ..."
fi
echo -e "' is running $resetvid"
if $strace
then
which xterm
if [ $? -eq 0 ]
then
pid=$(head -n1 "$vtfile" | sed 's/^ *//' | tr -s ' ' '\t' | cut -f 2)
echo "checking pid=$pid; quit with 'ctrl + c' in the xterm window"
xterm -title "'strace' checks '$1'" 2> /dev/null -e sudo strace -f -p $pid
else
echo "Please install 'xterm' for this function to work"
exit
fi
fi
else
inpath=$(which "$1")
if [ "$inpath" == "" ]
then
echo -e "$redback no path found to '$1' $resetvid"
else
echo -e "$blueback '$1' is not running $resetvid"
fi
fi
rm -r "$tmpdir"
检查Lubuntu中的终端窗口(LXTerminal作为x-terminal-emulator
和自定义gnome-terminal
窗口启动),
$ running -v -s term
----- try with: ----------------------------------------------------------------
UID PID PPID C STIME TTY TIME CMD
sudodus 2087 1384 0 13:33 ? 00:00:00 x-terminal-emulator
sudodus 2108 1269 0 13:33 ? 00:00:17 /usr/lib/gnome-terminal/gnome-terminal-server
--------------------------------------------------------------------------------
no path found to 'term'
$ running -v -s x-terminal-emulator
ps -ef: active:
UID PID PPID C STIME TTY TIME CMD
sudodus 2087 1384 0 13:33 ? 00:00:00 x-terminal-emulator
'x-terminal-emulator' is running
/usr/bin/xterm
checking pid=2087; quit with 'ctrl + c' in the xterm window
有很多活动一旦光标位于终端窗口。
正在启动grep
(等待来自的输入/dev/stdin
)
$ grep -i --color 'hello'
asdf
Hello World
Hello World
检查一下
$ running -s grep
ps -ef: active:
'grep ...' is running
/usr/bin/xterm
checking pid=14982; quit with 'ctrl + c' in the xterm window
没有太多活动,您可以确定正在发生的事情。
iotop
,如果进程繁忙,CPU使用率可能不一定是一个指标。用C编写并经过优化的程序可能使用最少的CPU。我用Python写的一些指示器安排了一个重复的任务来运行,因此它可能会使用CPU短暂地更新指示器菜单,然后坐在那里。
strace
方法更好,但是可能没有必要或不可用。
不知道您是否仍然需要它,但是仍然要知道一个有用的技巧:如果程序似乎在没有任何输出的情况下退出,则可以通过执行以下命令检查它是否在后台运行
ps -efa | grep "program_name"
干杯!
如果您在终端(例如终端仿真器或典型的ssh会话)中运行Shell,则您的Shell几乎可以肯定地启用了作业控制。在大多数情况下,这使获得问题的答案非常容易。
键入Ctrl+Z以暂停该过程,然后bg
在后台继续执行该过程,然后在外壳程序中键入空行,以便它检查程序是否被信号停止。
如果该进程尝试从终端读取,它将立即收到SIGTTIN
信号并被挂起。(启用作业控制后,系统一次仅允许从终端读取一个进程。)外壳程序将报告此情况。然后,您可以键入fg
以在前台继续该过程,然后按常规输入要由程序读取的输入。
mp@ubuntu:~$ sleep 30 # a program that is not reading from the terminal
^Z
[1]+ Stopped sleep 30
mp@ubuntu:~$ bg
[1]+ sleep 30 &
mp@ubuntu:~$
mp@ubuntu:~$
mp@ubuntu:~$ cat - # a program that is reading from the terminal
^Z
[1]+ Stopped cat -
mp@ubuntu:~$ bg
[1]+ cat - &
mp@ubuntu:~$
[1]+ Stopped cat -
mp@ubuntu:~$ jobs -l
[1]+ 3503 Stopped (tty input) cat -
mp@ubuntu:~$ fg
cat -
hi
hi
某些程序(例如编辑器)将捕获或忽略由Ctrl+Z终端生成的信号,或者将终端置于控制字符甚至不生成信号的模式。你需要使用在这种情况下更先进的技术,如使用strace
,看看进程正在做read
,select
,poll
,等。
PS1
提示您。