In Ruby, local variables are defined by the parser when it first encounters an assignment, and are then in scope from that point on.
Here's a little demonstration:
foo # NameError: undefined local variable or method `foo' for main:Object
if false
foo = 42
end
foo # => nil
As you can see, the local variable does exist on line 7 even though the assignment on line 4 was never executed. It was, however, parsed and that's why the local variable foo
exists. But because the assignment was never executed, the variable is uninitialized and thus evaluates to nil
and not 42
.
In Ruby, most uninitialized or even non-existing variables evaluate to nil
. This is true for local variables, instance variables and global variables:
defined? foo #=> nil
local_variables #=> []
if false
foo = 42
end
defined? foo #=> 'local-variable'
local_variables #=> [:foo]
foo #=> nil
foo.nil? #=> true
defined? @bar #=> nil
instance_variables #=> []
@bar #=> nil
@bar.nil? #=> true
# warning: instance variable @bar not initialized
defined? $baz #=> nil
$baz #=> nil
# warning: global variable `$baz' not initialized
$baz.nil? #=> true
# warning: global variable `$baz' not initialized
It is, however, not true for class hierarchy variables and constants:
defined? @@wah #=> nil
@@wah
# NameError: uninitialized class variable @@wah in Object
defined? QUUX #=> nil
QUUX
# NameError: uninitialized constant Object::QUUX
This is a red herring:
defined? fnord #=> nil
local_variables #=> []
fnord
# NameError: undefined local variable or method `fnord' for main:Object
The reason why you get an error here is not that unitialized local variables don't evaluate to nil
, it is that fnord
is ambiguous: it could be either an argument-less message send to the default receiver (i.e. equivalent to self.fnord()
) or an access to the local variable fnord
.
In order to disambiguate that, you need to add a receiver or an argument list (even if empty) to tell Ruby that it is a message send:
self.fnord
# NoMethodError: undefined method `fnord' for main:Object
fnord()
# NoMethodError: undefined method `fnord' for main:Object
or make sure that the parser (not the evaluator) parses (not executes) an assignment before the usage, to tell Ruby that it is a local variable:
if false
fnord = 42
end
fnord #=> nil
And, of course, nil
is an object (it is the only instance of class NilClass
) and thus has an object_id
method.