在命令行上,我输入了一个命令并按回车键。它没有输出任何内容。我如何判断它是正在运行但尚未输出,还是正在要求用户输入?
答案1
有几种方法:
尝试发出输入结束信号:如果没有超级用户权限,很难知道引擎盖下发生了什么。可以做的是按Ctrl+ d。终端和实用程序典范模式在收到与此组合键绑定的 EOT 信号后,将所有可用文本发送到
read()
syscall,如果没有输入 -read()
返回负退出状态,大多数实用程序都会将其接受为退出信号。因此,如果实用程序正在等待输入,它将在收到组合键后退出。否则,该实用程序要么正在运行任务,要么编写不正确。监视系统调用:如果您有超级用户权限,则可以
strace
在另一个终端中运行以查看当前正在执行的操作。为此,您需要找出程序的 PID。例如,在另一个终端选项卡中运行pgrep -f firefox
which may 1234 作为示例,然后运行sudo strace -f -p 1234
。如果您看到的输出停留在read()
syscall 上,则意味着该命令可能正在等待输入。否则,如果您看到 syscall 正在运行,则命令正在执行其他操作。查看相关问题用来strace
判断长时间运行的命令是否已经退出。使用命令自己的方法:除其他外,诸如
dd
使用信号之类的实用程序。例如,如果您使用kill -USR1 1234
(其中 1234 是正在运行的dd
命令的 PID),它将打印到 stdout 当前处理的字节数。当然,这首先需要了解命令的这种行为。上述两种方法更为通用,不需要深入了解每个命令的行为(尽管最好始终了解您实际正在执行的内容 - 否则您可能会冒着运行可能造成损害的命令的风险)。
答案2
如何判断程序正在运行或需要用户输入
这取决于程序以及您如何调用它。
通常但并非总是会出现一个提示,表明程序正在要求输入。
如果你不确定,你可以检查程序的进程是否繁忙
使用 CPU - 使用
top
或者htop
读取或写入 - 使用
sudo iotop -o
当程序运行完成后,你会看到 shell 的提示符。
Shell脚本running
我有一个用于检查程序是否正在运行的 shellscript,现在我添加了当发现 ... 时-s
使其运行的选项(根据 Sergiy Kolodyazhnyy 的回答)。sudo strace -f -p <PID>
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
shellscript 代码
#!/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
没有太多活动,您可以识别正在发生的事情。
答案3
如果您在终端(例如终端仿真器或典型的 ssh 会话)中运行 shell,则您的 shell 几乎肯定已启用作业控制。在大多数情况下,这使得您能够非常轻松地获得问题的答案。
键入Ctrl+Z以暂停该进程,然后bg
在后台继续该进程,然后在 shell 中键入一个空行,以便它检查该程序是否被信号停止。
如果进程试图从终端读取数据,它将立即收到信号SIGTTIN
并被暂停。(启用作业控制后,系统一次只允许一个进程从终端读取数据。)shell 将报告此情况。然后您可以键入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
等。
答案4
为什么不简单地查看 top 命令的 S(状态)列呢?如果您的程序正在等待输入,则很有可能它处于休眠状态而不是运行状态,这意味着 top 将输出 S(表示这次处于休眠状态)而不是 R(表示正在运行)。尝试使用此命令作为示例:
top -b -n 1 | sed -n '7,12p' | awk '{printf "%6s %-10s %-4s %-s\n",$1,$2,$8,$NF}'