红宝石救援区 - 只用一个命令回应

时间:2014-02-08 17:12:25

标签: ruby eval next rescue redo

我正在运行一个经常超时的API脚本。当发生这种情况时,我正在使用begin/rescue块将其发送到redo,但是在运行redo命令之前想要记录命令行中发生的事情。

begin
#...api query...
rescue ErrorClass
  puts("retrying #{id}") && redo
end

不幸的是,上面的脚本不起作用。只运行第一个命令。

我想强制救援块像这样运行多行代码:

begin
 # api query
rescue ErrorClass do ###or:# rescue ErrorClass do |e|
  puts "retrying #{id}"
  redo
 end

但那些也不起作用。

我很幸运创建了一个单独的方法来运行:

def example
  id = 34314
  begin
    5/0
  rescue ZeroDivisionError
    eval(handle_zerodiv_error(id))
  end
end

def handle_zerodiv_error(id)
  puts "retrying #{id}"
  "redo"
end

......实际上有效。但是在我看来,它需要太多的代码行,而且根据我的导师使用eval并不是犹太人。

2 个答案:

答案 0 :(得分:2)

使用&&do使您无关紧要。 &&版本不起作用,因为puts返回nil,因此通过&&的快捷方式评估,不会评估要遵循的部分。如果您改为使用||;,那么它将起作用:

begin
  ...
rescue ErrorClass
  puts("retrying #{id}") || redo
end

begin
  ...
rescue ErrorClass
  puts("retrying #{id}"); redo
end

但即使这样也没有必要。你似乎在某种程度上认为你需要rescue内的一个块来写多行,但这没有意义,因为你没有使用单行的块。没有Ruby构造只有在有多行时才需要块。所以,只需将它们分成多行:

begin
  ...
rescue ErrorClass
  puts("retrying #{id}")
  redo
end

答案 1 :(得分:1)

内置retry。此示例来自“The Ruby Programming Language”第162页。

require  "open-uri"

tries = 0
begin
  tries +=1
  open("http://www.example.com/"){|f| puts f.readlines}
rescue OpenURI::HTTPError => e
  puts e.message
  if (tries < 4)
    sleep (2**tries)  # wait for 2, 4 or 8 seconds
    retry             # and try again
  end
end