如何判断命令是否正在运行或正在等待用户输入?


14

在命令行上,我键入了一个命令,然后按Enter。它不输出任何东西。我如何知道它是否正在运行但尚未输出,或者正在要求用户输入?


如果正在等待,则不会PS1提示您。
Prvt_Yadav '19

1.请告诉我们它是哪个程序(没有声音),我们可以为您提供更精确的建议,包括预期内容和检查方法;2.当程序最终要求输入或已完成时,您是否想收到警报(以便将某些内容写入终端窗口?
sudodus

如果期望输入,我会假设您收到提示,并在其前面提示输入信息。
Rinzwind

5
@Rinzwind-这是一个糟糕的假设。我想到的第一个反例是cat命令。只需cat自行键入,它将等待来自stdin的输入,但不给出提示。许多其他命令的行为类似,因为它们期望从stdin或文件输入,但不区分不同的输入源(交互式终端,管道,文件...)。
Dave Sherohman

Answers:


15

有几种方法:

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

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

  3. 使用命令自己的方法:除其他外,诸如dd使用信号之类的实用程序。例如,如果您使用kill -USR1 1234(其中1234是正在运行的dd命令的PID ),它将打印到stdout当前正在处理的字节数。当然,这首先需要了解命令的这种行为。上面的两种方法比较通用,不需要深入了解每个命令的行为(尽管最好总是知道您实际执行的是什么,否则您就有可能运行可能会造成损坏的命令)。


+1。感谢您的strace方法:-)但是,更简单的方法也很有用(对于每个程序通用或特定)。其中一些没有超级用户权限就可以工作。示例:检查是否dd正在做某事,并检查为什么grep --color asdf静默等待。
sudodus

@sudodus啊,很好的提醒dd我将其添加。
Sergiy Kolodyazhnyy

您不需要超级用户特权即可调试用户拥有的进程。好吧,除非您没有正确设置系统
罗斯兰

6

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

它取决于程序以及如何调用它。

  • 通常但并非总是会有提示,表明程序正在请求输入。

  • 如果不确定,可以检查程序进程是否繁忙

    • 使用CPU-使用tophtop

    • 读取或写入-使用 sudo iotop -o

  • 程序完成后,您将看到外壳程序的提示。

Shell脚本 running

我有一个shellscript,用于检查程序是否正在运行,现在我添加了一个选项,-ssudo strace -f -p <PID>在找到...时使其运行(根据Sergiy Kolodyazhnyy的回答)。

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

Shell脚本代码

#!/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短暂地更新指示器菜单,然后坐在那里。
Sergiy Kolodyazhnyy

@SergiyKolodyazhnyy,是的,您是正确的。该strace方法更好,但是可能没有必要或不可用。
sudodus

同意 我不认为它是预装Ubuntu的,可能已经过时了。
瑟吉·科洛迪亚兹尼

1

不知道您是否仍然需要它,但是仍然要知道一个有用的技巧:如果程序似乎在没有任何输出的情况下退出,则可以通过执行以下命令检查它是否在后台运行

ps -efa | grep "program_name"

干杯!


1

如果您在终端(例如终端仿真器或典型的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,看看进程正在做readselectpoll,等。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.