Misunderstanding scope can cause problems in your application. Watch this lesson to learn how Python scope works and the hidden implications it presents. Local scope, nonlocal scope, and global scope variables are demonstrated and explained.
Man: In Python, there are always at least three nested scopes whose name spaces are directly available. To show you what that means, I'm going to create a function called Who Am I. Inside, I'm going to create a variable named I and have it just be a string that says, "I am Groot." Then we'll print out the string I and color function so I can execute that with the Python command, and it prints out I am Groot.
Let's create a nested function and redefine I. I'm going to create a new function inside of the Who Am I function called local Groot. Inside of that, I'm going to create a variable named I and have it say, "I am local Groot." Then we'll print the value of I. We'll execute our local Groot method, and then we'll print I again.
I'll save that, execute it. When it executes, it says, "I am Groot," both times. It's like nothing even happened.
Let me prove to you that it actually did change. To do that, I'm going to print the value of I while we're inside the function. I'll save that, run it again. We define the value of I. We print I. It says, "I am Groot." We execute our local Groot method then print out I am local Groot. Then we print out the value of I again where I still remains I am Groot.
Let's do one more thing where I show you that. I'm going to print out the ID of the variable I which will show us the object ID for that in Python, so we'll do that inside the local Groot method. We'll do it here in our outer scope. Then we'll do it again one more time after we've called our local Groot function. We set the value to I.
We printed the ID which is the 7360, printed out the value I am Groot. Then we went inside our local Groot function, printed the value of I, and that has a different value. It's 3768. So that tells us that it's a completely different variable even though it has the same variable name. After we exit that function, we print ID again. That again refers to the value of the variable I in the outer scope.
I can get rid of those print commands real quick just because It will make our explanation a little easier as I start to tell you about what if you want to override that variable of I? We can do that in Python using the non-local keyword. I'm going to create another function in here. I'm going to call this one non-local Groot.
Inside of it, I'm going to define a variable with the non-local keyword and give it the variable name I, set our value. This time we'll say, "I am non-local Groot."
We'll call our non-local function and then we'll print out our value for the variable I. Everything behaved as it did before except after I called that non-local Groot function, when we printed out the value of I, it was no longer the value of I from this outer scope. It was the value I am non-local Groot.
What happens is whenever you use the non-local keyword, you're telling that method that the variable I is not local to this function, and it should look in the outer scope for a variable with that same name. That's what happened here. When non-local Groot executed, it found this variable I in the outer scope. Then when we printed that out, we got the value I am non-local Groot.
Finally, we have global scopes. We'll create another function here. We'll call this one global Groot. Inside of it, I'm going to use the global keyword to define the variable I. We'll set I equal to the value of I am global Groot. Go down to the bottom here. We're going to execute our global Groot function or method, print out our value of I.
When we run this, it says, "I'm non-local Groot." You might have been expecting that to say, "I am global Groot," because we defined I globally.
Let me add one more print line down here, and I'll show you why that didn't work. Save that, run it again, now finally we get our I am global Groot print statement. Let me tell you what happened here. When we called global Groot, we had previously called non-local Groot, and that set the value of the non-local I to I am non-local Groot.
Enclose this so we have a little more room here. The global keyword indicates that the variable lives in global scope. The non-local keyword indicates that the variable lives in the enclosing scope.
When we call print for the variable I here on line 21, we were still in the enclosed scope of the Who Am I function, so that's why it printed out I am non-local Groot. It wasn't until we got down here to line 24 and we printed the variable I outside of any other scope that we finally allowed the global scope to take precedence and print out I am global Groot.
If that seems confusing, it probably should. It's the reason that Python best practices recommend against using globally scoped variables. Once you have several functions or several methods nested within each other, it can be really difficult to tell where a global variable takes precedence and where it doesn't.
If you find yourself tempted to use a global variable, what I would recommend instead is creating a parameter for your method that allows you to get around that and then pass in the value that you need. We'll pass in the name and then we'll return that. Get rid of this extra print statement down here, save that. We need to pass in our name to global Groot.
Now when that executes, we passed in the parameter that we needed, we avoid the need for a global variable, and then we just return the value that we needed from that method.