3

由于我的一些草率编码,我注意到未声明的实例变量似乎评估 nil 而未声明的局部变量没有。实例变量的这个默认 nill 值是我可以利用的预期行为(例如使用条件检查变量是否已设置为真)还是这只是一个应该不理会的怪癖?

2.6.6 :001 > puts @test

 => nil 
2.6.6 :002 > puts test
Traceback (most recent call last):
        2: from (irb):2
        1: from (irb):2:in `test'
ArgumentError (wrong number of arguments (given 0, expected 2..3))
4

2 回答 2

5

根据 Ruby 文档:

ruby 的实例变量不需要声明。这意味着对象的灵活结构。事实上,每个实例变量在第一次被引用时都被动态地附加到一个对象上。

https://ruby-doc.org/docs/ruby-doc-bundle/UsersGuide/rg/instancevars.html

您还可以获得所有先前定义的实例变量的列表:

ClassName.instance_variables

或者

a = ClassName.new 
a.instance_variables #specific to instance
于 2021-01-13T21:06:24.147 回答
3

tl;dr是的,这记录在Assignment中。

一个事物是否被定义,它被定义为什么,可以用defined?关键字来表示。

2.6.5 :003 > defined?(fnord)
 => nil 
2.6.5 :004 > fnord = 42
 => 42 
2.6.5 :005 > defined?(fnord)
 => "local-variable" 
2.6.5 :006 > defined?($fnord)
 => nil 
2.6.5 :007 > $fnord = 42
 => 42 
2.6.5 :008 > defined?($fnord)
 => "global-variable" 
2.6.5 :009 > defined?(@fnord)
 => nil 
2.6.5 :010 > @fnord = 42
 => 42 
2.6.5 :011 > defined?(@fnord)
 => "instance-variable" 
2.6.5 :012 > defined?(FNORD)
 => nil 
2.6.5 :013 > FNORD = 42
 => 42 
2.6.5 :014 > defined?(FNORD)
 => "constant" 

这对于调试很有用,但我不鼓励在应用程序代码中“利用”它。


首先,让我们澄清一下您的示例。test是一种方法main。它继承自Kernel#test

2.6.5 :004 > defined?(test)
 => "method" 
2.6.5 :005 > method(:test)
 => #<Method: main.test> 

一个合适的例子看起来像这样。

2.6.5 :008 > puts @fnord

 => nil 
2.6.5 :009 > puts fnord
Traceback (most recent call last):
        4: from /Users/schwern/.rvm/rubies/ruby-2.6.5/bin/irb:23:in `<main>'
        3: from /Users/schwern/.rvm/rubies/ruby-2.6.5/bin/irb:23:in `load'
        2: from /Users/schwern/.rvm/rubies/ruby-2.6.5/lib/ruby/gems/2.6.0/gems/irb-1.0.0/exe/irb:11:in `<top (required)>'
        1: from (irb):9
NameError (undefined local variable or method `fnord' for main:Object)

局部变量

该错误暗示正在发生的事情:模棱两可。fnord可以是局部变量fnordself.fnord. 两者都没有被声明,Ruby 不会猜到你的意思,所以你会得到一个 NameError。

在 Ruby 中,局部变量名和方法名几乎相同。如果您没有指定这些模棱两可的名称之一,ruby 将假定您希望调用一个方法。一旦您指定了名称,ruby 将假定您希望引用一个局部变量。

可以使用local_variables发现声明的局部变量。

2.6.5 :012 > foo = 42
 => 42 
2.6.5 :013 > Kernel.local_variables
 => [:foo, :_] 

实例变量

未初始化的实例变量的值为 nil。如果您在启用警告的情况下运行 Ruby,则在访问未初始化的实例变量时会收到警告。

有多种方法可以内省实例变量,例如instance_variablesinstance_variables_defined?。虽然在某些情况下编写库时可能需要这样做,但我强烈反对nil在应用程序代码中利用和“未定义”之间的区别;它使事情变得脆弱。

类变量

访问未初始化的类变量将引发 NameError 异常。

全局变量

未初始化的全局变量的值为 nil。

于 2021-01-13T21:28:40.047 回答