如何判断程序正在运行或需要用户输入

如何判断程序正在运行或需要用户输入

在命令行上,我输入了一个命令并按回车键。它没有输出任何内容。我如何判断它是正在运行但尚未输出,还是正在要求用户输入?

答案1

有几种方法:

  1. 尝试发出输入结束信号:如果没有超级用户权限,很难知道引擎盖下发生了什么。可以做的是按Ctrl+ d。终端和实用程序典范模式在收到与此组合键绑定的 EOT 信号后,将所有可用文本发送到read()syscall,如果没有输入 -read()返回负退出状态,大多数实用程序都会将其接受为退出信号。因此,如果实用程序正在等待输入,它将在收到组合键后退出。否则,该实用程序要么正在运行任务,要么编写不正确。

  2. 监视系统调用:如果您有超级用户权限,则可以strace在另一个终端中运行以查看当前正在执行的操作。为此,您需要找出程序的 PID。例如,在另一个终端选项卡中运行pgrep -f firefoxwhich may 1234 作为示例,然后运行sudo strace -f -p 1234​​。如果您看到的输出停留在read()syscall 上,则意味着该命令可能正在等待输入。否则,如果您看到 syscall 正在运行,则命令正在执行其他操作。查看相关问题用来strace判断长时间运行的命令是否已经退出。

  3. 使用命令自己的方法:除其他外,诸如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找到一些程序
  • 如果您希望,也可以stracexterm窗口中进行。

    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来查看进程是否正在执行readselectpoll等。

答案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}'

相关内容