3

我需要为块内发生的所有异常添加调试信息,但我不想弄乱回溯。$!在 1.9.3 中似乎不允许这样做;raise无论我尝试什么,都在替换回溯。

想法?

这是我最初使用的:

def self.load(filename, virtual_path = nil)
  t = Template.new(filename, virtual_path)
  t.is_page? ? Page.new(t) : t
rescue
  raise $!, "Error loading template '#{filename}'#{virtual_path ? " under virtual path '" + virtual_path  + "'" : ""}: #{$!}"
end

到目前为止我发现的最好的是:

def self.load(filename, virtual_path = nil)
  t = Template.new(filename, virtual_path)
  t.is_page? ? Page.new(t) : t
rescue => e
    raise e, "Error loading template '#{filename}'#{virtual_path ? " under virtual path '" + virtual_path  + "'" : ""}: #{e.message} #{e.backtrace}"
end

这会将原始堆栈跟踪转储到消息中,但仍不会将旧堆栈跟踪保留为堆栈跟踪

4

1 回答 1

4

如果您查看Kernel#raise方法,它可以采用三个参数:

raise(exception [, string [, array]])

如果要保留回溯,则应指定array参数,即回调信息。

例子:

假设您最初拥有:

def some_method()
    raise('original message')
end

some_method
#=> scratch.rb:10:in `some_method': original message (RuntimeError)
#   from scratch.rb:16:in `<main>'

您可以使用异常的第三个参数来引发具有更新消息和相同回溯的新异常:

def some_method()
    begin
        raise('error message')
    rescue
        raise $!, 'new message', $!.backtrace
    end
end

some_method
#=> scratch.rb:10:in `some_method': new message (RuntimeError)
#       from scratch.rb:16:in `<main>'

如您所见,新异常与原始异常相同,只是更新了消息。

于 2013-03-13T13:07:27.870 回答