__main__ and scoping in python
from:https://stackoverflow.com/questions/4775579/main-and-scoping-in-python
I was somehow surprised by the following behavior:
def main():
print "%s" % foo
if __name__ == "__main__":
foo = "bar"
main()
i.e. a module function has access to enclosing variables in the __main__
. What's the explanation for it?
Variables in the current modules global scope are visible everywhere in the module -- this rule also holds for the __main__
module.
From Guido's tutorial:
At any time during execution, there are at least three nested scopes whose namespaces are directly accessible:
- the innermost scope, which is searched first, contains the local names
- the scopes of any enclosing functions, which are searched starting with the nearest enclosing scope, contains non-local, but also non-global names
- the next-to-last scope contains the current module’s global names
- the outermost scope (searched last) is the namespace containing built-in names
-
__main__
module? – user225312 Jan 23 '11 at 18:58 -
@AA: The main script is treated by Python as a module with the name
__main__
. You even can doimport __main__
(but usually that's a Bad Idea). – Sven Marnach Jan 23 '11 at 19:16
The thing here is that:
if __name__ == "__main__":
foo = "bar"
defines a global variable named foo in that script. so any function of that module will have access to it.
The piece of code listed above is global to the module and not inside any function.
foo is a module global variable (it's not in any function). All scopes within the module can access it.
In python there's the global scope, and functions have their own scopes. So it you define foo under the name==main, it's in the global scope. Also, it's not a mistake to use a variable which hasn't been declared yet, in a function, if it will be declared by the time the function will be called.
__main__
. There's anif
that happens to compare a variable that happens to be called__name__
to something that happens to be the string literal"__main__"
. – user395760 Jan 23 '11 at 18:53__main__
is a module, and each module has an associated scope. Tryimport __main__; type(__main__)
in the interpreter (not in IPython). – Sven Marnach Jan 23 '11 at 21:48if
introduces a new scope - which would be a more pressing issue. (Admittedly, "There is no scope__main__
is misleading, strictly speaking) – user395760 Jan 23 '11 at 21:50