Ruby学习笔记(14)_异常
来源:互联网 发布:win7服务优化工具 编辑:程序博客网 时间:2024/06/08 10:19
异常和执行总是被联系在一起。如果您打开一个不存在的文件,且没有恰当地处理这种情况,那么您的程序则被认为是低质量的。
如果异常发生,则程序停止。异常用于处理各种类型的错误,这些错误可能在程序执行期间发生,所以要采取适当的行动,而不至于让程序完全停止。
Ruby 提供了一个完美的处理异常的机制。我们可以在 begin/end 块中附上可能抛出异常的代码,并使用 rescue 子句告诉 Ruby 完美要处理的异常类型。
语法:
begin #开始 raise.. #抛出异常rescue [ExceptionType = StandardException] #捕获指定类型的异常 缺省值是StandardException $! #表示异常信息 $@ #表示异常出现的代码位置else #其余异常 ..ensure #不管有没有异常,进入该代码块end #结束
从 begin 到 rescue 中的一切是受保护的。如果代码块执行期间发生了异常,控制会传到 rescue 和 end 之间的块。
对于 begin 块中的每个 rescue 子句,Ruby 把抛出的异常与每个参数进行轮流比较。如果 rescue 子句中命名的异常与当前抛出的异常类型相同,或者是该异常的父类,则匹配成功。
如果异常不匹配所有指定的错误类型,我们可以在所有的 rescue 子句后使用一个 else 子句。
实例:
#!/usr/bin/rubybegin file = open("/unexistant_file") if file puts "File opened successfully" endrescue file = STDINendprint file, "==", STDIN, "\n"
以上实例运行输出结果为。您可以看到,STDIN 取代了 file ,因为打开失败。
#<IO:0xb7d16f84>==#<IO:0xb7d16f84>
使用 retry 语句
可以使用 rescue 块捕获异常,然后使用 retry 语句从开头开始执行 begin 块。
语法:
begin # 这段代码抛出的异常将被下面的 rescue 子句捕获rescue # 这个块将捕获所有类型的异常 retry # 这将把控制移到 begin 的开头end
实例:
#!/usr/bin/rubybegin file = open("/unexistant_file") if file puts "File opened successfully" endrescue fname = "existant_file" retryend
以下是处理流程:
打开时发生异常。
跳到 rescue。fname 被重新赋值。
通过 retry 跳到 begin 的开头。
这次文件成功打开。
继续基本的过程。
注意:如果被重新命名的文件不存在,本实例代码会无限尝试。所以异常处理时,谨慎使用 retry。
使用 raise 语句
您可以使用 raise 语句抛出异常。下面的方法在调用时抛出异常。它的第二个消息将被输出。
语法:
raise 或raise "Error Message" 或raise ExceptionType, "Error Message"或raise ExceptionType, "Error Message" condition
- 第一种形式简单地重新抛出当前异常(如果没有当前异常则抛出一个 RuntimeError)。这用在传入异常之前需要解释异常的异常处理程序中。
- 第二种形式创建一个新的 RuntimeError 异常,设置它的消息为给定的字符串。该异常之后抛出到调用堆栈。
- 第三种形式使用第一个参数创建一个异常,然后设置相关的消息为第二个参数。
- 第四种形式与第三种形式类似,您可以添加任何额外的条件语句(比如 unless)来抛出异常。
实例:
#!/usr/bin/rubybegin 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/rubybegin 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 块始终都会运行。
(就是java的 try..catch..finally)
语法:
begin #.. 过程 #.. 抛出异常rescue #.. 处理错误 ensure #.. 最后确保执行 #.. 这总是会执行end
实例:
begin raise 'A test exception.'rescue Exception => e puts e.message puts e.backtrace.inspectensure puts "Ensuring execution"end
结果:
A test exception.
[“main.rb:4”]
Ensuring execution
使用 else 语句
如果提供了 else 子句,它一般是放置在 rescue 子句之后,任意 ensure 之前。
else 子句的主体只有在代码主体没有抛出异常时执行。
实例:
begin # 抛出 'A test exception.' puts "I'm not raising exception"rescue Exception => e puts e.message puts e.backtrace.inspectelse 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 后匹配将被执行的 catchend或throw :lablename condition#.. 这不会被执行catch :lablename do#.. 在遇到一个 throw 后匹配将被执行的 catchend
实例:
def promptAndGet(prompt) print prompt res = readline.chomp throw :quitRequested if res == "!" return resendcatch :quitRequested do name = promptAndGet("Name: ") age = promptAndGet("Age: ") sex = promptAndGet("Sex: ") # .. # 处理信息endpromptAndGet("Name:")
- Ruby学习笔记(14)_异常
- Ruby学习笔记_异常处理rescue
- Ruby学习笔记_单元测试
- Ruby学习笔记_对象过程
- Ruby学习笔记(03)_变量
- Ruby学习笔记(04)_判断
- Ruby学习笔记(05)_循环
- Ruby学习笔记(07)_块
- Ruby学习笔记(08)_模块Module
- Ruby学习笔记(09)_数组
- Ruby学习笔记(12)_范围Range
- Ruby学习笔记(13)_迭代器
- Ruby学习笔记(16)_面对对象
- Ruby学习笔记(18)_冒号用法
- Ruby学习笔记(19)_继承
- Ruby学习笔记(20)_符号Symbol
- Ruby学习笔记(21)_限制权限
- ruby学习笔记(9)--异常处理
- Nginx启用, 停止, 平滑重启命令
- Solovay-Strassen素性判定----C语言
- Hibernate配置文件之映射文件配置(一)
- Elasticsearch命令(curl)大全
- Linux shell sed命令在文件行首行尾添加字符
- Ruby学习笔记(14)_异常
- 谈谈深度学习中的 Batch_Size Batch_Size(批尺寸)是机器学习中一个重要参数,涉及诸多矛盾,下面逐一展开。 首先,为什么需要有 Batch_Size 这个参数? Batch 的选
- Ajax表格传值
- 我必须得告诉大家的MySQL优化原理
- java 二进制数据与16进制字符串相互转化
- 继承中的原型链
- Math.abs( x )
- 在对话框文档上点击右键后添加的信息
- CHAPTER 1 Using neural nets to recognize handwritten digits