How references to variables are resolved in Python
Solution 1:
In an ideal world, you'd be right and some of the inconsistencies you found would be wrong. However, CPython has optimized some scenarios, specifically function locals. These optimizations, together with how the compiler and evaluation loop interact and historical precedent, lead to the confusion.
Python translates code to bytecodes, and those are then interpreted by a interpreter loop. The 'regular' opcode for accessing a name is LOAD_NAME
, which looks up a variable name as you would in a dictionary. LOAD_NAME
will first look up a name as a local, and if that fails, looks for a global. LOAD_NAME
throws a NameError
exception when the name is not found.
For nested scopes, looking up names outside of the current scope is implemented using closures; if a name is not assigned to but is available in a nested (not global) scope, then such values are handled as a closure. This is needed because a parent scope can hold different values for a given name at different times; two calls to a parent function can lead to different closure values. So Python has LOAD_CLOSURE
, MAKE_CLOSURE
and LOAD_DEREF
opcodes for that situation; the first two opcodes are used in loading and creating a closure for a nested scope, and the LOAD_DEREF
will load the closed-over value when the nested scope needs it.
Now, LOAD_NAME
is relatively slow; it will consult two dictionaries, which means it has to hash the key first and run a few equality tests (if the name wasn't interned). If the name isn't local, then it has to do this again for a global. For functions, that can potentially be called tens of thousands of times, this can get tedious fast. So function locals have special opcodes. Loading a local name is implemented by LOAD_FAST
, which looks up local variables by index in a special local names array. This is much faster, but it does require that the compiler first has to see if a name is a local and not global. To still be able to look up global names, another opcode LOAD_GLOBAL
is used. The compiler explicitly optimizes for this case to generate the special opcodes. LOAD_FAST
will throw an UnboundLocalError
exception when there is not yet a value for the name.
Class definition bodies on the other hand, although they are treated much like a function, do not get this optimization step. Class definitions are not meant to be called all that often; most modules create classes once, when imported. Class scopes don't count when nesting either, so the rules are simpler. As a result, class definition bodies do not act like functions when you start mixing scopes up a little.
So, for non-function scopes, LOAD_NAME
and LOAD_DEREF
are used for locals and globals, and for closures, respectively. For functions, LOAD_FAST
, LOAD_GLOBAL
and LOAD_DEREF
are used instead.
Note that class bodies are executed as soon as Python executes the class
line! So in example 1, class B
inside class A
is executed as soon as class A
is executed, which is when you import the module. In example 2, C
is not executed until f()
is called, not before.
Lets walk through your examples:
-
You have nested a class
A.B
in a classA
. Class bodies do not form nested scopes, so even though theA.B
class body is executed when classA
is executed, the compiler will useLOAD_NAME
to look upx
.A.B().f()
is a function (bound to theB()
instance as a method), so it usesLOAD_GLOBAL
to loadx
. We'll ignore attribute access here, that's a very well defined name pattern. -
Here
f().C.z
is at class scope, so the functionf().C().g()
will skip theC
scope and look at thef()
scope instead, usingLOAD_DEREF
. -
Here
var
was determined to be a local by the compiler because you assign to it within the scope. Functions are optimized, soLOAD_FAST
is used to look up the local and an exception is thrown. -
Now things get a little weird.
class A
is executed at class scope, soLOAD_NAME
is being used.A.x
was deleted from the locals dictionary for the scope, so the second access tox
results in the globalx
being found instead;LOAD_NAME
looked for a local first and didn't find it there, falling back to the global lookup.Yes, this appears inconsistent with the documentation. Python-the-language and CPython-the implementation are clashing a little here. You are, however, pushing the boundaries of what is possible and practical in a dynamic language; checking if
x
should have been a local inLOAD_NAME
would be possible but takes precious execution time for a corner case that most developers will never run into. -
Now you are confusing the compiler. You used
x = x
in the class scope, and thus you are setting a local from a name outside of the scope. The compiler findsx
is a local here (you assign to it), so it never considers that it could also be a scoped name. The compiler usesLOAD_NAME
for all references tox
in this scope, because this is not an optimized function body.When executing the class definition,
x = x
first requires you to look upx
, so it usesLOAD_NAME
to do so. Nox
is defined,LOAD_NAME
doesn't find a local, so the globalx
is found. The resulting value is stored as a local, which happens to be namedx
as well.print x
usesLOAD_NAME
again, and now finds the new localx
value. -
Here you did not confuse the compiler. You are creating a local
y
,x
is not local, so the compiler recognizes it as a scoped name from parent functionf2().myfunc()
.x
is looked up withLOAD_DEREF
from the closure, and stored iny
.
You could see the confusion between 5 and 6 as a bug, albeit one that is not worth fixing in my opinion. It was certainly filed as such, see issue 532860 in the Python bug tracker, it has been there for over 10 years now.
The compiler could check for a scoped name x
even when x
is also a local, for that first assignment in example 5. Or LOAD_NAME
could check if the name is meant to be a local, really, and throw an UnboundLocalError
if no local was found, at the expense of more performance. Had this been in a function scope, LOAD_FAST
would have been used for example 5, and an UnboundLocalError
would be thrown immediately.
However, as the referenced bug shows, for historical reasons the behaviour is retained. There probably is code out there today that'll break were this bug fixed.
Solution 2:
In two words, the difference between example 5 and example 6 is that in example 5 the variable x
is also assigned to in the same scope, while not in example 6. This triggers a difference that can be understood by historical reasons.
This raises UnboundLocalError:
x = "foo"
def f():
print x
x = 5
f()
instead of printing "foo". It makes a bit of sense, even if it seems strange at first: the function f() defines the variable x
locally, even if it is after the print, and so any reference to x
in the same function must be to that local variable. At least it makes sense in that it avoids strange surprizes if you have by mistake reused the name of a global variable locally, and are trying to use both the global and the local variable. This is a good idea because it means that we can statically know, just by looking at a variable, which variable it means. For example, we know that print x
refers to the local variable (and thus may raise UnboundLocalError) here:
x = "foo"
def f():
if some_condition:
x = 42
print x
f()
Now, this rule doesn't work for class-level scopes: there, we want expressions like x = x
to work, capturing the global variable x
into the class-level scope. This means that class-level scopes don't follow the basic rule above: we can't know if x
in this scope refers to some outer variable or to the locally-defined x
--- for example:
class X:
x = x # we want to read the global x and assign it locally
bar = x # but here we want to read the local x of the previous line
class Y:
if some_condition:
x = 42
print x # may refer to either the local x, or some global x
class Z:
for i in range(2):
print x # prints the global x the 1st time, and 42 the 2nd time
x = 42
So in class scopes, a different rule is used: where it would normally raise UnboundLocalError --- and only in that case --- it instead looks up in the module globals. That's all: it doesn't follow the chain of nested scopes.
Why not? I actually doubt there is a better explanation that "for historical reasons". In more technical terms, it could consider that the variable x
is both locally defined in the class scope (because it is assigned to) and should be passed in from the parent scope as a lexically nested variable (because it is read). It would be possible to implement it by using a different bytecode than LOAD_NAME
that looks up in the local scope, and falls back to using the nested scope's reference if not found.
EDIT: thanks wilberforce for the reference to http://bugs.python.org/issue532860. We may have a chance to get some discussion reactivated with the proposed new bytecode, if we feel that it should be fixed after all (the bug report considers killing support for x = x
but was closed for fear of breaking too much existing code; instead what I'm suggesting here would be to make x = x
work in more cases). Or I may be missing another fine point...
EDIT2: it seems that CPython did precisely that in the current 3.4 trunk: http://bugs.python.org/issue17853 ... or not? They introduced the bytecode for a slightly different reason and don't use it systematically...
Solution 3:
Long story short, this is a corner case of Python's scoping that is a bit inconsistent, but has to be kept for backwards compatibility (and because it's not that clear what the right answer should be). You can see lots of the original discussion about it on the Python mailing list when PEP 227 was being implemented, and some in the bug for which this behaviour is the fix.
We can work out why there's a difference using the dis
module, which lets us look inside code objects to see the bytecode a piece of code has been compiled to. I'm on Python 2.6, so the details of this might be slightly different - but I see the same behaviour, so I think it's probably close enough to 2.7.
The code that initialises each nested MyClass
lives in a code object that you can get to via the attributes of the top-level functions. (I'm renaming the functions from example 5 and example 6 to f1
and f2
respectively.)
The code object has a co_consts
tuple, which contains the myfunc
code object, which in turn has the code that runs when MyClass
gets created:
In [20]: f1.func_code.co_consts
Out[20]: (None,
'x in f2',
<code object myfunc at 0x1773e40, file "<ipython-input-3-6d9550a9ea41>", line 4>)
In [21]: myfunc1_code = f1.func_code.co_consts[2]
In [22]: MyClass1_code = myfunc1_code.co_consts[3]
In [23]: myfunc2_code = f2.func_code.co_consts[2]
In [24]: MyClass2_code = myfunc2_code.co_consts[3]
Then you can see the difference between them in bytecode using dis.dis
:
In [25]: from dis import dis
In [26]: dis(MyClass1_code)
6 0 LOAD_NAME 0 (__name__)
3 STORE_NAME 1 (__module__)
7 6 LOAD_NAME 2 (x)
9 STORE_NAME 2 (x)
8 12 LOAD_NAME 2 (x)
15 PRINT_ITEM
16 PRINT_NEWLINE
17 LOAD_LOCALS
18 RETURN_VALUE
In [27]: dis(MyClass2_code)
6 0 LOAD_NAME 0 (__name__)
3 STORE_NAME 1 (__module__)
7 6 LOAD_DEREF 0 (x)
9 STORE_NAME 2 (y)
8 12 LOAD_NAME 2 (y)
15 PRINT_ITEM
16 PRINT_NEWLINE
17 LOAD_LOCALS
18 RETURN_VALUE
So the only difference is that in MyClass1
, x
is loaded using the LOAD_NAME
op, while in MyClass2
, it's loaded using LOAD_DEREF
. LOAD_DEREF
looks up a name in an enclosing scope, so it gets 'x in myfunc'. LOAD_NAME
doesn't follow nested scopes - since it can't see the x
names bound in myfunc
or f1
, it gets the module-level binding.
Then the question is, why does the code of the two versions of MyClass
get compiled to two different opcodes? In f1
the binding is shadowing x
in the class scope, while in f2
it's binding a new name. If the MyClass
scopes were nested functions instead of classes, the y = x
line in f2
would be compiled the same, but the x = x
in f1
would be a LOAD_FAST
- this is because the compiler would know that x
is bound in the function, so it should use the LOAD_FAST
to retrieve a local variable. This would fail with an UnboundLocalError
when it was called.
In [28]: x = 'x in module'
def f3():
x = 'x in f2'
def myfunc():
x = 'x in myfunc'
def MyFunc():
x = x
print x
return MyFunc()
myfunc()
f3()
---------------------------------------------------------------------------
Traceback (most recent call last)
<ipython-input-29-9f04105d64cc> in <module>()
9 return MyFunc()
10 myfunc()
---> 11 f3()
<ipython-input-29-9f04105d64cc> in f3()
8 print x
9 return MyFunc()
---> 10 myfunc()
11 f3()
<ipython-input-29-9f04105d64cc> in myfunc()
7 x = x
8 print x
----> 9 return MyFunc()
10 myfunc()
11 f3()
<ipython-input-29-9f04105d64cc> in MyFunc()
5 x = 'x in myfunc'
6 def MyFunc():
----> 7 x = x
8 print x
9 return MyFunc()
UnboundLocalError: local variable 'x' referenced before assignment
This fails because the MyFunc
function then uses LOAD_FAST
:
In [31]: myfunc_code = f3.func_code.co_consts[2]
MyFunc_code = myfunc_code.co_consts[2]
In [33]: dis(MyFunc_code)
7 0 LOAD_FAST 0 (x)
3 STORE_FAST 0 (x)
8 6 LOAD_FAST 0 (x)
9 PRINT_ITEM
10 PRINT_NEWLINE
11 LOAD_CONST 0 (None)
14 RETURN_VALUE
(As an aside, it's not a big surprise that there should be a difference in how scoping interacts with code in the body of classes and code in a function. You can tell this because bindings at the class level aren't available in methods - method scopes aren't nested inside the class scope in the same way as nested functions are. You have to explicitly reach them via the class, or by using self.
(which will fall back to the class if there's not also an instance-level binding).)