Является ли SystemExit особым видом исключения?

Как SystemExit ведет себя иначе, чем другие Exception s? Думаю, я понимаю некоторые аргументы в пользу того, почему было бы неправильно поднимать правильное Исключение. Например, вам не хотелось бы, чтобы это произошло так странно:

begin
  exit
rescue => e
  # Silently swallow up the exception and don't exit
end

Но как игнорирует rescue SystemExit? (Какие критерии он использует?)

Ответ 1

Когда вы пишете rescue без одного или нескольких классов, это то же самое, что и запись:

begin
  ...
rescue StandardError => e
  ...
end

Однако существуют исключения, которые не наследуются от StandardError. SystemExit является одним из них, и поэтому он не захватывается. Вот поднабор иерархии в Ruby 1.9.2, который вы можете узнать сами:

BasicObject
  Exception
    NoMemoryError
    ScriptError
      LoadError
        Gem::LoadError
      NotImplementedError
      SyntaxError
    SecurityError
    SignalException
      Interrupt
    StandardError
      ArgumentError
      EncodingError
        Encoding::CompatibilityError
        Encoding::ConverterNotFoundError
        Encoding::InvalidByteSequenceError
        Encoding::UndefinedConversionError
      FiberError
      IOError
        EOFError
      IndexError
        KeyError
        StopIteration
      LocalJumpError
      NameError
        NoMethodError
      RangeError
        FloatDomainError
      RegexpError
      RuntimeError
      SystemCallError
      ThreadError
      TypeError
      ZeroDivisionError
    SystemExit
    SystemStackError
    fatal

Таким образом вы можете записать только SystemExit с помощью:

begin
  ...
rescue SystemExit => e
  ...
end

... или вы можете выбрать для каждого исключения исключение, включая SystemExit:

begin
  ...
rescue Exception => e
  ...
end

Попробуйте сами:

begin
  exit 42
  puts "No no no!"
rescue Exception => e
  puts "Nice try, buddy."
end
puts "And on we run..."

#=> "Nice try, buddy."
#=> "And on we run..."

Обратите внимание, что этот пример не будет работать (некоторые версии?) IRB, который предоставляет свой собственный метод exit, который маскирует нормальный объект # exit.

В 1.8.7:

method :exit
#=> #<Method: Object(IRB::ExtendCommandBundle)#exit>

В 1.9.3:

method :exit
#=> #<Method: main.irb_exit>

Ответ 2

Простой пример:

begin
  exit
  puts "never get here"
rescue SystemExit
  puts "rescued a SystemExit exception"
end

puts "after begin block"

Также можно прочитать выход status/success? и т.д.:

begin
  exit 1
rescue SystemExit => e
  puts "Success? #{e.success?}" # Success? false
end

begin
  exit
rescue SystemExit => e
  puts "Success? #{e.success?}" # Success? true
end

Полный список методов: [:status, :success?, :exception, :message, :backtrace, :backtrace_locations, :set_backtrace, :cause]