Ruby "defined?"运算符工作错误?

标签 ruby exception behavior defined

所以,我们有代码:

class Foo
  def bar
    puts "Before existent: #{(defined? some_variable)}"
    puts "Before not_existent: #{(defined? nonexistent_variable)}"

    raise "error"

    some_variable = 42
  rescue
    puts "exception"
  ensure
    puts "Ensure existent: #{(defined? some_variable)}"
    puts "Ensure not_existent: #{(defined? nonexistent_variable)}"
  end
end

然后从 irb 调用它:

> Foo.new.bar

并且,这将返回:

Before existent:
Before not_existent:
exception
Ensure existent: local-variable
Ensure not_existent:
=> nil

现在是问题 - 为什么?我们在 之前引发异常 some_variable 被定义。 为什么会这样?为什么 some_variable 定义在 ensure block 中? (顺便说一句,它定义为零)

更新: 感谢@Max 的回答,但是如果我们更改代码以使用实例变量:

class Foo
  def bar
    puts "Before existent: #{(defined? @some_variable)}"
    puts "Before not_existent: #{(defined? @nonexistent_variable)}"

    raise "error"

    @some_variable = 42
  ensure
    puts "Ensure existent: #{(defined? @some_variable)}"
    puts "Ensure not_existent: #{(defined? @nonexistent_variable)}"
  end
end

它按预期工作:

Before existent:
Before not_existent:
Ensure existent:
Ensure not_existent:

为什么?

最佳答案

首先要注意的是 defined? 是一个关键字,而不是一个方法。这意味着是 specially handled在构建语法树时由解析器在编译期间(就像 ifreturnnext 等)而不是动态查找运行时。

这就是为什么 defined? 可以处理通常会引发错误的表达式:defined?(what is this even) #=> nil 因为解析器可以排除它的来自正常评估过程的参数。

真正令人困惑的一点是,尽管它是一个关键字,但它的行为却是still determined at runtime。 .它使用解析器魔术来确定其参数是否为实例变量、常量、方法等。然后调用普通的 Ruby 方法来确定这些特定类型是否已在运行时定义:

// ...
case DEFINED_GVAR:
if (rb_gvar_defined(rb_global_entry(SYM2ID(obj)))) {
    expr_type = DEFINED_GVAR;
}
break;
case DEFINED_CVAR:
// ...
if (rb_cvar_defined(klass, SYM2ID(obj))) {
    expr_type = DEFINED_CVAR;
}
break;
case DEFINED_CONST:
// ...
if (vm_get_ev_const(th, klass, SYM2ID(obj), 1)) {
    expr_type = DEFINED_CONST;
}
break;
// ...

rb_cvar_defined 函数与 Module#class_variable_defined? 调用的函数相同,例如。

所以 defined? 很奇怪。真的很奇怪。它的行为可能会因参数而异,我什至不敢打赌它在不同的 Ruby 实现中是相同的。基于此,我建议不要使用它,而是尽可能使用 Ruby 的 *_defined? 方法。

关于Ruby "defined?"运算符工作错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29371966/

相关文章:

android - 如何禁用 BottomNavigationView 点击和触摸?

ruby-on-rails - 我可以在做种时将我的所有产品更新给特定用户吗?

ruby-on-rails - 提交 ROE 后似乎无法保留表单数据

ruby-on-rails - 使用帮助程序在 Rails 3 应用程序中打印根 URL

angular - Observable.catch() 和 RxJ 中的异常处理

plone - 有没有办法扩展 Plone Dexterity 的 INameFromTitle 行为?

ruby-on-rails - 如何在 Ruby 中只读取 URL 的 header ?

java - 使用时异常? : operator in Java 8 lambda expression

java - 奇怪的 ClassCastException

symfony1 - symfony : actAs: { Timestampable: ~ }