我正在运行一个经常超时的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并不是犹太人。
答案 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