如何从Ruby检查具有特定pid的进程是否正在运行?

Pis*_*tos 45 ruby pid process

如果有多种方式,请列出.我只知道一个,但我想知道是否有一个更干净的Ruby内部方式.

小智 59

这些Process.getpgidProcess::kill方法之间的区别似乎是当pid存在但是由另一个用户拥有时会发生的情况.Process.getpgid将返回一个答案,Process::kill将抛出异常(Errno::EPERM).

基于此,我建议Process.getpgid,如果只是因为它可以使您免于捕获两个不同的异常.

这是我使用的代码:

begin
  Process.getpgid( pid )
  true
rescue Errno::ESRCH
  false
end
Run Code Online (Sandbox Code Playgroud)

  • oneliner: `(Process.getpgid(pid) 救援 nil).present?` (3认同)

Dus*_*tin 47

如果它是您希望"拥有"的进程(例如,您使用此进程来验证您控制的进程的pid),则可以向其发送sig 0.

>> Process.kill 0, 370
=> 1
>> Process.kill 0, 2
Errno::ESRCH: No such process
    from (irb):5:in `kill'
    from (irb):5
>> 
Run Code Online (Sandbox Code Playgroud)

  • 不幸的是,如果进程处于"僵尸"状态,这也会返回1,因此不会告诉您进程是否已经死亡,只是系统上是否存在具有该PID的进程(例如,如果您有一个打开的流到一个进程,即使进程已终止,它也将返回1).另请注意,操作系统可以重复使用PID ... (7认同)
  • dang,打败了我.我看到"新答案已被张贴"的事情,我的心沉了:( (3认同)

Pis*_*tos 46

@John T,@ Dustin:实际上,伙计们,我仔细阅读了过程rdocs,它看起来像

Process.getpgid( pid )
Run Code Online (Sandbox Code Playgroud)

应用相同技术是一种不那么暴力的手段.

  • 我认为这里的含义是"杀死0"是"暴力的" - 这是一个错误.文档说:"如果sig为0,则不发送信号,但仍会执行错误检查;这可用于检查是否存在进程ID或进程组ID." (3认同)

bal*_*alu 27

对于子进程,其他解决方案(如发送信号)将不会按预期运行:它们将指示进程在实际退出时仍在运行.

如果要检查自己生成的进程,可以使用Process.waitpid.如果您正在使用该Process::WNOHANG标志,nil则不会阻止该调用,只要子进程未退出,该调用将被返回.

例:

pid = Process.spawn('sleep 5')
Process.waitpid(pid, Process::WNOHANG) # => nil
sleep 5
Process.waitpid(pid, Process::WNOHANG) # => pid
Run Code Online (Sandbox Code Playgroud)

如果pid不属于子进程,则抛出异常(Errno::ECHILD: No child processes).

这同样适用于Process.waitpid2.

  • 我发现这比上面提到的解决方案更可靠.谢谢! (2认同)

kom*_*tsu 6

这就是我一直在做的事情:

def alive?(pid)
  !!Process.kill(0, pid) rescue false
end
Run Code Online (Sandbox Code Playgroud)


小智 5

您可以尝试使用

Process::kill 0, pid
Run Code Online (Sandbox Code Playgroud)

其中 pid 是 pid 编号,如果 pid 正在运行,则应返回 1。