雷林鹏分享:Ruby 异常

  Ruby 异常

  异常和执行总是被联系在一起。如果您打开一个不存在的文件,且没有恰当地处理这种情况,那么您的程序则被认为是低质量的。

  如果异常发生,则程序停止。异常用于处理各种类型的错误,这些错误可能在程序执行期间发生,所以要采取适当的行动,而不至于让程序完全停止。

  Ruby 提供了一个完美的处理异常的机制。我们可以在 begin/end 块中附上可能抛出异常的代码,并使用 rescue 子句告诉 Ruby 完美要处理的异常类型。

  语法

  begin

  # -

  rescue OneTypeOfException

  # -

  rescue AnotherTypeOfException

  # -

  else

  # 其他异常

  ensure

  # 总是被执行

  end

  从 begin 到 rescue 中的一切是受保护的。如果代码块执行期间发生了异常,控制会传到 rescue 和 end 之间的块。

  对于 begin 块中的每个 rescue 子句,Ruby 把抛出的异常与每个参数进行轮流比较。如果 rescue 子句中命名的异常与当前抛出的异常类型相同,或者是该异常的父类,则匹配成功。

  如果异常不匹配所有指定的错误类型,我们可以在所有的 rescue 子句后使用一个 else 子句。

  实例

  #!/usr/bin/ruby

  begin

  file = open("/unexistant_file")

  if file

  puts "File opened successfully"

  end

  rescue

  file = STDIN

  end

  print file, "==", STDIN, "\n"

  这将产生以下结果。您可以看到,STDIN 取代了 file ,因为打开失败。

  #==#

  使用 retry 语句

  您可以使用 rescue 块捕获异常,然后使用 retry 语句从开头开始执行 begin 块。

  语法

  begin

  # 这段代码抛出的异常将被下面的 rescue 子句捕获

  rescue

  # 这个块将捕获所有类型的异常

  retry # 这将把控制移到 begin 的开头

  end

  实例

  #!/usr/bin/ruby

  begin

  file = open("/unexistant_file")

  if file

  puts "File opened successfully"

  end

  rescue

  fname = "existant_file"

  retry

  end

  以下是处理流程:

  打开时发生异常。

  跳到 rescue。fname 被重新赋值。

  通过 retry 跳到 begin 的开头。

  这次文件成功打开。

  继续基本的过程。

  注意:如果被重新命名的文件不存在,本势力代码会无限尝试。所以异常处理时,谨慎使用 retry。

  使用 raise 语句

  您可以使用 raise 语句抛出异常。下面的方法在调用时抛出异常。它的第二个消息将被输出。

  语法

  raise

  OR

  raise "Error Message"

  OR

  raise ExceptionType, "Error Message"

  OR

  raise ExceptionType, "Error Message" condition

  第一种形式简单地重新抛出当前异常(如果没有当前异常则抛出一个 RuntimeError)。这用在传入异常之前需要解释异常的异常处理程序中。

  第二种形式创建一个新的 RuntimeError 异常,设置它的消息为给定的字符串。该异常之后抛出到调用堆栈。

  第三种形式使用第一个参数创建一个异常,然后设置相关的消息为第二个参数。

  第四种形式与第三种形式类似,您可以添加任何额外的条件语句(比如 unless)来抛出异常。

  实例

  #!/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.

  另一个演示 raise 用法的实例:

  #!/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"]

  使用 ensure 语句

  有时候,无论是否抛出异常,您需要保证一些处理在代码块结束时完成。例如,您可能在进入时打开了一个文件,当您退出块时,您需要确保关闭文件。

  ensure 子句做的就是这个。ensure 放在最后一个 rescue 子句后,并包含一个块终止时总是执行的代码块。它与块是否正常退出、是否抛出并处理异常、是否因一个未捕获的异常而终止,这些都没关系,ensure 块始终都会运行。

  语法

  begin

  #.. 过程

  #.. 抛出异常

  rescue

  #.. 处理错误

  ensure

  #.. 最后确保执行

  #.. 这总是会执行

  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 子句之后,任意 ensure 之前。

  else 子句的主体只有在代码主体没有抛出异常时执行。

  语法

  begin

  #.. 过程

  #.. 抛出异常

  rescue

  #.. 处理错误

  else

  #.. 如果没有异常则执行

  ensure

  #.. 最后确保执行

  #.. 这总是会执行

  end

  实例

  begin

  # 抛出 ‘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

  使用 $! 变量可以捕获抛出的错误消息。

  Catch 和 Throw

  raise 和 rescue 的异常机制能在发生错误时放弃执行,有时候需要在正常处理时跳出一些深层嵌套的结构。此时 catch 和 throw 就派上用场了。

  catch 定义了一个使用给定的名称(可以是 Symbol 或 String)作为标签的块。块会正常执行知道遇到一个 throw。

  语法

  throw :lablename

  #.. 这不会被执行

  catch :lablename do

  #.. 在遇到一个 throw 后匹配将被执行的 catch

  end

  OR

  throw :lablename condition

  #.. 这不会被执行

  catch :lablename do

  #.. 在遇到一个 throw 后匹配将被执行的 catch

  end

  实例

  下面的实例中,如果用户键入 ‘!‘ 回应任何提示,使用一个 throw 终止与用户的交互。

  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: ")

  # ..

  # 处理信息

  end

  promptAndGet("Name:")

  上面的程序需要人工交互,您可以在您的计算机上进行尝试。这将产生以下结果:

  Name: Ruby on Rails

  Age: 3

  Sex: !

  Name:Just Ruby

  类 Exception

  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

  # 写出数据 ...

  rescue

  # 发生错误

  raise FileSaveError.new($!)

  end

  end

  在这里,最重要的一行是 raise FileSaveError.new($!)。我们调用 raise 来示意异常已经发生,把它传给 FileSaveError 的一个新的实例,由于特定的异常引起数据写入失败。

  (编辑:雷林鹏 来源:网络)

原文地址:https://www.cnblogs.com/pengpeng1208/p/10168105.html

时间: 2024-10-13 09:35:47

雷林鹏分享:Ruby 异常的相关文章

雷林鹏分享:Ruby 多线程

Ruby 多线程 每个正在系统上运行的程序都是一个进程.每个进程包含一到多个线程. 线程是程序中一个单一的顺序控制流程,在单个程序中同时运行多个线程完成不同的工作,称为多线程. Ruby 中我们可以通过 Thread 类来创建多线程,Ruby的线程是一个轻量级的,可以以高效的方式来实现并行的代码. 创建 Ruby 线程 要启动一个新的线程,只需要调用 Thread.new 即可: # 线程 #1 代码部分 Thread.new { # 线程 #2 执行代码 } # 线程 #1 执行代码 实例 以

雷林鹏分享:Ruby 哈希(Hash)

Ruby 哈希(Hash) 哈希(Hash)是类似 "employee" => "salary" 这样的键值对的集合.哈希的索引是通过任何对象类型的任意键来完成的,而不是一个整数索引,其他与数组相似. 通过键或值遍历哈希的顺序看起来是随意的,且通常不是按照插入顺序.如果您尝试通过一个不存在的键访问哈希,则方法会返回 nil. 创建哈希 与数组一样,有各种不同的方式来创建哈希.您可以通过 new 类方法创建一个空的哈希: months = Hash.new 您也

雷林鹏分享:Ruby 字符串(String)

Ruby 字符串(String) Ruby 中的 String 对象存储并操作一个或多个字节的任意序列,通常表示那些代表人类语言的字符. 最简单的字符串是括在单引号(单引号字符)内.在引号标记内的文本是字符串的值: 'This is a simple Ruby string literal' 如果您需要在单引号字符串内使用单引号字符,那么需要在单引号字符串使用反斜杠,这样 Ruby 解释器就不会认为这个单引号字符会终止字符串: 'Won\'t you read O\'Reilly\'s book

雷林鹏分享:Ruby JSON

Ruby JSON 本章节我们将为大家介绍如何使用 Ruby 语言来编码和解码 JSON 对象. 环境配置 在使用 Ruby 编码或解码 JSON 数据前,我们需要先安装 Ruby JSON 模块.在安装该模块前你需要先安装 Ruby gem,我们使用 Ruby gem 安装 JSON 模块. 但是,如果你使用的是最新版本的 Ruby,可能已经安装了 gem,解析来我们就可以使用以下命令来安装Ruby JSON 模块: $gem install json 使用 Ruby 解析 JSON 以下为J

雷林鹏分享:Ruby CGI方法

Ruby CGI方法 以下为CGI类的方法列表: 序号方法描述 1CGI::new([ level="query"]) 创建 CGI 对象.query可以是以下值: query: 没有 HTML 生成输出 html3: HTML3.2 html4: HTML4.0 Strict html4Tr: HTML4.0 Transitional html4Fr: HTML4.0 Frameset 2CGI::escape( str) 使用 URL 编码来转义字符串 3CGI::unescape

雷林鹏分享:Ruby 面向对象

Ruby 面向对象 Ruby 是纯面向对象的语言,Ruby 中的一切都是以对象的形式出现.Ruby 中的每个值都是一个对象,即使是最原始的东西:字符串.数字,甚至连 true 和 false 都是对象.类本身也是一个对象,是 Class 类的一个实例.本章将向您讲解所有与 Ruby 面向对象相关的主要功能. 类用于指定对象的形式,它结合了数据表示法和方法,把数据整理成一个整齐的包.类中的数据和方法被称为类的成员. Ruby 类定义 当您定义一个类时,您实际是定义了一个数据类型的蓝图.这实际上并没

雷林鹏分享:Ruby File 类和方法

Ruby File 类和方法 File 表示一个连接到普通文件的 stdio 对象.open 为普通文件返回该类的一个实例. 类方法 序号方法 & 描述 1File::atime( path) 返回 path 的最后访问时间. 2File::basename( path[, suffix]) 返回 path 末尾的文件名.如果指定了 suffix,则它会从文件名末尾被删除. 例如:File.basename("/home/users/bin/ruby.exe") #=>

雷林鹏分享:Ruby 文件的输入与输出

Ruby 文件的输入与输出 Ruby 提供了一整套 I/O 相关的方法,在内核(Kernel)模块中实现.所有的 I/O 方法派生自 IO 类. 类 IO 提供了所有基础的方法,比如 read. write. gets. puts. readline. getc 和 printf. 本章节将讲解所有 Ruby 中可用的基础的 I/O 函数.如需了解更多的函数,请查看 Ruby 的 IO 类. puts 语句 在前面的章节中,您赋值给变量,然后使用 puts 语句打印输出. puts 语句指示程序

雷林鹏分享:Ruby 日期 &amp; 时间(Date &amp; Time)

Ruby 日期 & 时间(Date & Time) Time 类在 Ruby 中用于表示日期和时间.它是基于操作系统提供的系统日期和时间之上.该类可能无法表示 1970 年之前或者 2038 年之后的日期. 本教程将让您熟悉日期和时间的所有重要的概念. 创建当前的日期和时间 下面是获取当前的日期和时间的简单实例: #!/usr/bin/ruby -w time1 = Time.new puts "Current Time : " + time1.inspect # Ti