• 国家国际发展合作署举行揭牌仪式 2018-04-21
  • 春潮涌动,千余青少年相约科技馆 2018-04-21
  • 英俄用“关闭电视台”互相威胁 双边关系雪上加霜 2018-04-21
  • 拔不拔赖清德?蔡英文何去何从 2018-04-21
  • 我国阅读指数首次发布 人均纸书阅读量为4.66本 2018-04-20
  • 《夏目的美丽日记》绿色度测评报告 2018-04-20
  • 只要打动人心,文艺片也可以很好看 2018-04-19
  • 《荒城纪》揭幕北京大学生电影节 2018-04-19
  • 贵州桐梓:困难群众打工差点错过“移民搬迁”之后 2018-04-19
  • 《英雄联盟》2018年季中冠军赛入围赛抽签将4月24日举行 2018-04-19
  • 全国最美志愿者黄梅生:孤儿的“校长爸爸” 2018-04-19
  • 《荒城纪》揭幕北京大学生电影节 2018-04-18
  • 贵州桐梓:困难群众打工差点错过“移民搬迁”之后 2018-04-18
  • 老布什夫人芭芭拉放弃治疗,余生将在家中“颐养” 2018-04-18
  • 音乐剧《朝暮有情人》演绎港穗两地三代情 2018-04-18
  • 【ruby教程】Ruby 异常

    栏目: Ruby 发布于: 2014-11-26 07:37:00

    Ruby 异常

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

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

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

    语法

    begin  
    # -  
    rescue OneTypeOfException  
    # -  
    rescue AnotherTypeOfException  
    # -  
    else  
    # 其他异常
    ensure
    # 总是被执行
    end
    

    beginrescue 中的一切是受?;さ?。如果代码块执行期间发生了异常,控制会传到 rescueend 之间的块。

    对于 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 ,因为打开失败。

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

    使用 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)作为标签的块??榛嵴V葱兄烙龅揭桓?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 的一个新的实例,由于特定的异常引起数据写入失败。


    本文转载自:w3cschool
    本站文章除注明转载外,均为本站原创或编译?;队魏涡问降淖?,但请务必注明出处。
    转载请注明:文章转载自 七星彩票平台
    本文标题:【ruby教程】Ruby 异常
    IT技术书籍推荐:
    PHP从入门到精通(第3版)
    PHP从入门到精通(第3版)
    明日科技 (作者)
    《PHP从入门到精通(第3版)》从初学者角度出发,通过通俗易懂的语言、丰富多彩的实例,详细介绍了使用PHP进行网络开发应该掌握的各方面技术。全书共分4篇25章,包括初识PHP、PHP环境搭建和开发工具、PHP语言基础、流程控制语句、字符串操作、正则表达式、PHP数组、PHP与Web页面交互、PHP与JavaScript交互、日期和时间、Cookie与Session、图形图像处理技术、文件系统、面向对象、PHP加密技术、MySQL数据库基础、phpMyAdmin图形化管理工具、PHP操作MySQL数据库、PDO数据库抽象层、ThinkPHP框架、Smarty模板技术、PHP与XML技术、PHP与Ajax技术、应用Smarty模板开发电子商务网站、应用ThinkPHP框架开发明日导航网等内容。书中所有知识都结合具体实例进行介绍,涉及的程序代码均附以详细的注释,可以使读者轻松领会PHP程序开发的精髓,快速提高开发技能。