5

如何获取外部命令的输出并从中提取值?

我有这样的事情:

stdin, stdout, stderr, wait_thr = Open3.popen3("#{path}/foobar", configfile)

if /exit 0/ =~ wait_thr.value.to_s
    runlog.puts("Foobar exited normally.\n")
    puts "Test completed."
    someoutputvalue = stdout.read("TX.*\s+(\d+)\s+")
    puts "Output value: " + someoutputvalue
end

我没有在标准输出上使用正确的方法,因为 Ruby 告诉我它不能将字符串转换为整数。

例如,如果输出是

"TX So and so:     28"

我只想得到“ 28”。我验证了上面的正则表达式匹配我需要匹配的内容,我只是想知道如何将提取的值存储在变量中。

这样做的正确方法是什么?我在文档中的任何地方都找不到标准输出可用的方法。我stout.read从 Ruby 1.9.3 开始使用。

4

1 回答 1

15

所需的所有信息都在Popen3 文档中,但您必须阅读所有内容并仔细查看示例。您还可以从Process 文档中收集有用的信息。

也许这会“更好地解释它:

require 'open3'

captured_stdout = ''
captured_stderr = ''
exit_status = Open3.popen3(ENV, 'date') {|stdin, stdout, stderr, wait_thr|
  pid = wait_thr.pid # pid of the started process.
  stdin.close
  captured_stdout = stdout.read
  captured_stderr = stderr.read
  wait_thr.value # Process::Status object returned.
}

puts "STDOUT: " + captured_stdout
puts "STDERR: " + captured_stderr
puts "EXIT STATUS: " + (exit_status.success? ? 'succeeded' : 'failed')

运行输出:

STDOUT: Wed Jun 12 07:07:12 MST 2013
STDERR:
EXIT STATUS: succeeded

注意事项:

  • 你经常不得不closestdin。如果被调用的应用程序期望 STDIN 上的输入,它将挂起,直到它看到流关闭,然后将继续其处理。
  • stdin, stdout,stderr是 IO 句柄,因此您必须阅读IO 类文档以了解可用的方法。
  • 您必须stdin使用puts, printor write, and reador getsfrom stdoutand输出stderr
  • exit_status不是字符串,它是 Process::Status 类的实例。你可以尝试从它的to_s版本中解析,但不要。而是使用访问器查看它返回的内容。
  • 我传入了ENV哈希,因此子程序可以访问父程序看到的整个环境。没有必要这样做;相反,如果您不希望孩子可以访问所有内容,则可以为孩子创建一个简化的环境,或者您可以通过更改值来扰乱孩子对环境的看法。
  • 问题中发布的代码stdout.read("TX.*\s+(\d+)\s+")是,嗯......胡说八道。我不知道你从哪里得到的,因为 Ruby 的IO#readIO.read的 IO 类中没有记录任何类似的东西。

capture3如果您不需要写入被调用代码的 STDIN,它会更容易使用:

require 'open3'

stdout, stderr, exit_status = Open3.capture3('date')

puts "STDOUT: " + stdout
puts "STDERR: " + stderr
puts "EXIT STATUS: " + (exit_status.success? ? 'succeeded' : 'failed')

哪个输出:

STDOUT: Wed Jun 12 07:23:23 MST 2013
STDERR:
EXIT STATUS: succeeded

使用正则表达式从字符串中提取值是很简单的,并且在Regexp 文档中有很好的介绍。从最后一个代码示例开始:

stdout[/^\w+ (\w+ \d+) .+ (\d+)$/]
puts "Today is: " + [$1, $2].join(' ')

哪个输出:

Today is: Jun 12 2013

那是使用String.[]非常灵活的方法。

另一种方法是使用“命名捕获”:

/^\w+ (?<mon_day>\w+ \d+) .+ (?<year>\d+)$/ =~ stdout
puts "Today is: #{ mon_day } #{ year }"

输出相同的东西。命名捕获的缺点是它们速度较慢,因为我认为有点方便。


"TX So and so: 28"[/\d+$/]
=> "28"
于 2013-06-12T14:07:53.613 回答