Ruby - исключения

Выполнение и исключение всегда идут вместе. Если вы открываете несуществующий файл, то, если вы не справились с этой ситуацией должным образом, ваша программа считается некачественной.

Программа останавливается, если возникает исключение. Таким образом, исключения используются для обработки различных типов ошибок, которые могут возникнуть во время выполнения программы, и принятия соответствующих мер вместо полной остановки программы.

Ruby предоставляет хороший механизм для обработки исключений. Мы заключаем код, который может вызвать исключение, в начальный / конечный блок и используем предложения rescue, чтобы сообщить Ruby типы исключений, которые мы хотим обработать.

Синтаксис

begin  
# -  
rescue OneTypeOfException  
# -  
rescue AnotherTypeOfException  
# -  
else  
# Other exceptions
ensure
# Always will be executed
end

Все, от начала до спасения , защищено. Если происходит исключение во время выполнения этого блока кода, управление передается в блок между спасательным и концом .

Для каждого спасательного пункта в начать блок, Ruby сравнивает поднятый Exception против каждого из параметров , в свою очередь. Сопоставление будет успешным, если исключение, указанное в предложении rescue, совпадает с типом текущего исключения или является суперклассом этого исключения.

В случае, если исключение не соответствует ни одному из типов ошибок , указанных, нам разрешено использовать другое положение после всех спасательных пунктов.

пример

#!/usr/bin/ruby

begin
   file = open("/unexistant_file")
   if file
      puts "File opened successfully"
   end
rescue
      file = STDIN
end
print file, "==", STDIN, "\n"

Это даст следующий результат. Вы можете видеть, что STDIN заменяется файлом из-за сбоя открытия .

#<IO:0xb7d16f84>==#<IO:0xb7d16f84>

Использование оператора повтора

Вы можете захватить исключение , используя спасательный блок , а затем использовать Повторить заявление для выполнения начать блок с самого начала.

Синтаксис

begin
   # Exceptions raised by this code will 
   # be caught by the following rescue clause
rescue
   # This block will capture all types of exceptions
   retry  # This will move control to the beginning of begin
end

пример

#!/usr/bin/ruby

begin
   file = open("/unexistant_file")
   if file
      puts "File opened successfully"
   end
rescue
   fname = "existant_file"
   retry
end

Ниже приведен поток процесса -

  • Исключение при открытии.
  • Отправился на помощь. fname было переназначено.
  • По повторной попытке перейти к началу начала.
  • На этот раз файл открывается успешно.
  • Продолжил основной процесс.

NOTE- Обратите внимание, что если файл с замененным именем не существует, этот пример кода повторяется бесконечно. Будьте осторожны, если вы используете повтор для процесса исключения.

Использование инструкции Raise

Вы можете использовать рейз заявление сгенерирует исключение. Следующий метод вызывает исключение при каждом вызове. Будет напечатано второе сообщение.

Синтаксис

raise 

OR

raise "Error Message" 

OR

raise ExceptionType, "Error Message"

OR

raise ExceptionType, "Error Message" condition

Первая форма просто повторно вызывает текущее исключение (или RuntimeError, если текущего исключения нет). Это используется в обработчиках исключений, которым необходимо перехватить исключение перед его передачей.

Вторая форма создает новое исключение RuntimeError , устанавливая для своего сообщения заданную строку. Затем это исключение поднимается в стеке вызовов.

Третья форма использует первый аргумент для создания исключения, а затем устанавливает для связанного сообщения второй аргумент.

Четвертая форма похожа на третью, но вы можете добавить любой условный оператор, например, если не вызвать исключение.

пример

#!/usr/bin/ruby

begin  
   puts 'I am before the raise.'  
   raise 'An error has occurred.'  
   puts 'I am after the raise.'  
rescue  
   puts 'I am rescued.'  
end  
puts 'I am after the begin block.'

Это даст следующий результат -

I am before the raise.  
I am rescued.  
I am after the begin block.

Еще один пример, показывающий использование повышения -

#!/usr/bin/ruby

begin  
   raise 'A test exception.'  
rescue Exception => e  
   puts e.message  
   puts e.backtrace.inspect  
end

Это даст следующий результат -

A test exception.
["main.rb:4"]

Использование выражения обеспечения

Иногда вам нужно гарантировать, что некоторая обработка будет выполнена в конце блока кода, независимо от того, было ли возбуждено исключение. Например, у вас может быть файл, открытый при входе в блок, и вам нужно убедиться, что он закрывается при выходе из блока.

Предложение обеспечения делает именно это. sure идет после последнего предложения rescue и содержит фрагмент кода, который всегда будет выполняться при завершении блока. Не имеет значения, выходит ли блок нормально, если он вызывает и спасает исключение, или если он завершается неперехваченным исключением, блок обеспечения будет запущен.

Синтаксис

begin 
   #.. process 
   #..raise exception
rescue 
   #.. handle error 
ensure 
   #.. finally ensure execution
   #.. This will always execute.
end

пример

begin
   raise 'A test exception.'
rescue Exception => e
   puts e.message
   puts e.backtrace.inspect
ensure
   puts "Ensuring execution"
end

Это даст следующий результат -

A test exception.
["main.rb:4"]
Ensuring execution

Использование оператора else

Если предложение else присутствует, оно идет после предложений rescue и до любого обеспечения .

Тело предложения else выполняется только в том случае, если основная часть кода не вызывает исключений.

Синтаксис

begin 
   #.. process 
   #..raise exception
rescue 
   # .. handle error
else
   #.. executes if there is no exception
ensure 
   #.. finally ensure execution
   #.. This will always execute.
end

пример

begin
   # raise 'A test exception.'
   puts "I'm not raising exception"
rescue Exception => e
   puts e.message
   puts e.backtrace.inspect
else
   puts "Congratulations-- no errors!"
ensure
   puts "Ensuring execution"
end

Это даст следующий результат -

I'm not raising exception
Congratulations-- no errors!
Ensuring execution

Сообщение об ошибке может быть записано с помощью $! переменная.

Поймай и брось

Хотя механизм исключения подъема и спасения отлично подходит для прекращения выполнения, когда что-то идет не так, иногда приятно иметь возможность выпрыгнуть из какой-то глубоко вложенной конструкции во время нормальной обработки. Вот здесь и пригодятся улов и бросок.

Улов определяет блок , который помечен с данным именем (которое может быть символ или строка). Блок выполняется нормально, пока не встретится бросок.

Синтаксис

throw :lablename
#.. this will not be executed
catch :lablename do
#.. matching catch will be executed after a throw is encountered.
end

OR

throw :lablename condition
#.. this will not be executed
catch :lablename do
#.. matching catch will be executed after a throw is encountered.
end

пример

В следующем примере бросок используется для прекращения взаимодействия с пользователем, если '!' вводится в ответ на любой запрос.

def promptAndGet(prompt)
   print prompt
   res = readline.chomp
   throw :quitRequested if res == "!"
   return res
end

catch :quitRequested do
   name = promptAndGet("Name: ")
   age = promptAndGet("Age: ")
   sex = promptAndGet("Sex: ")
   # ..
   # process information
end
promptAndGet("Name:")

Вы должны попробовать вышеуказанную программу на своем компьютере, потому что она требует ручного вмешательства. Это даст следующий результат -

Name: Ruby on Rails
Age: 3
Sex: !
Name:Just Ruby

Исключение класса

Стандартные классы и модули Ruby вызывают исключения. Все классы исключений образуют иерархию с классом Exception вверху. Следующий уровень содержит семь разных типов -

  • Interrupt
  • NoMemoryError
  • SignalException
  • ScriptError
  • StandardError
  • SystemExit

Есть еще одно исключение на этом уровне, Fatal, но интерпретатор Ruby использует это только для внутренних целей.

И ScriptError, и StandardError имеют несколько подклассов, но нам не нужно вдаваться в подробности здесь. Важно то, что если мы создаем наши собственные классы исключений, они должны быть подклассами либо класса Exception, либо одного из его потомков.

Давайте посмотрим на пример -

class FileSaveError < StandardError
   attr_reader :reason
   def initialize(reason)
      @reason = reason
   end
end

Теперь посмотрите на следующий пример, в котором будет использоваться это исключение -

File.open(path, "w") do |file|
begin
   # Write out the data ...
rescue
   # Something went wrong!
   raise FileSaveError.new($!)
end
end

Важная строка здесь - поднять FileSaveError.new ($!) . Мы вызываем raise, чтобы сигнализировать о возникновении исключения, передавая ему новый экземпляр FileSaveError, причина которого в том, что это конкретное исключение привело к сбою записи данных.