Skip to content

Latest commit

 

History

History
188 lines (167 loc) · 5.91 KB

File metadata and controls

188 lines (167 loc) · 5.91 KB

Scope

Scope is the context in which a variable name is valid and can be used.​

There are several types of variables in Ruby, including local, instance, and class variables, all of which you've seen at this point. You are declaring a local variable anytime you write something like this: ​

some_var = 'some value'

​ The other types of variables mentioned have a special symbol at the beginning denoting them as such: ​

  • Instance variables begin with @
  • Class variables begin with @@

This reading is going to focus on local variables - we'll discuss instance and class variables more later, when we delve deeper into OOP. ​ Let's start with the basics - we can't use a variable before it is defined: ​

def pow(base, exponent)
  i = 0

  while i < exponent
    result = result * base  # Error: result is being used before it has been defined.

    i += 1
  end

  result
end

Block Scoping

When we open up a block, variables declared within the block will not be accessible once we exit the block. However, if we have declared a variable before we enter the block, we will have access to that variable inside the block. In fact, any changes made to that variable within the block will persist once we've exited the block. Let's see a few examples: ​

3.times do
  x = 4
end

puts x # Error - undefined local variable `x`

Because x is defined inside the block (we might call this the inner scope), we do not have access to it once we exit the block (in the outer or top-level scope). ​

x = 2

3.times do
  x += 2
end

puts x # outputs 8 (2 + 3*2)

This code works because we declare x before we enter the block. Within the block we modify x, and these changes persist after we have exited block, so when we puts x we see its modified value of 8.

The rule here is as follows: Local variables declared in outer scopes are available, and may be modified, in all inner scopes. Local variables declared in inner scopes are not available in outer scopes.

If/While statements

Keep in mind that if/while statements are different than blocks in Ruby, and these rules do not apply. Even though our code inside if/while statements is nested, this code is still treated as part of the outer scope. For example, we can do something like this: ​

if true
  i = 2
end

puts i # outputs 2

Or this: ​

while true
  i = 2
  break
end

puts i # outputs 2

Both of these code snippets will run without error even though it appears that i is being defined in an inner scope and accessed in an outer scope. Don't let the nesting fool you - the context does not change within these code sinppets!

Scope gates

We've discussed blocks and if/while statement, but what happens when we define a new method or class? The rules change - whenever we define a method, class, or module (you'll learn about those in the near future), we open up what is commonly referred to as a scope gate. What this means is that we lose access to all local variables previously defined and enter a brand new context. Let's see some examples: ​

x = 2

def display_x
  puts x
end

display_x # Error - undefined local variable `x`

Even though x is defined before we define the method and in what we might consider an "outer scope", the method definition opens up a new scope gate and we lose access to all previously defined local variables. We only have access to local variables declared within the method and the parameters of the method. To make the above work we would need to refactor it to take in a parameter: ​

x = 2

def display_x(x)
  puts x
end

display_x(x) # outputs 2

The above code is able to run without any problems. Note that while we name the method parameter the same as the variable we pass to the method, they are distinct variables that live in separate scopes. To be more clear, here is that same example with a clearer differentiation of the variables: ​

my_var = 2

def display_thing(method_var)
  puts method_var
end

display_thing(my_var) # outputs 2

Methods calling other methods

Methods != Local Variables

Based on what we've discussed so far, you may be wondering how we can call methods within other methods, given that every method opens up its own scope gate. The answer is that methods behave differently than local variables. Methods defined in the top level scope are accessible within other methods also defined in the top level scope. The short of it is that these methods actually get defined on an object called main and you are able to call them for the same reason you can call instance methods from other instance methods in a class. At this stage, don't worry too much about how exactly this is happening

  • just understand that this is the behavior you can expect. ​
def fourth_power(i)
  square(i) * square(i)
end

def square(i)
  i * i
end

fourth_power(2) # outputs 16 (2^4)

The above code is able to run without error. It's also worth noting that square is used in fourth_power despite being defined below it. Since we don't call fourth_power until the very last line, we have access to both methods. ​

Global variables

NOTE: This last bit about global variables is not essential.

While you shouldn't typically create global variables, you can do so with the $ prefix. Once a global variable has been declared, it is available in every scope in your program. Unless you absolutely have to (and you probably don't) you should avoid creating global variables as they are dangerous, and often unnecessary. ​

Additional Reading

We've covered some of the core principles of scope in Ruby. For a deeper dive on scope check out this resource.