Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How does Python importing exactly work?

I have two specific situations where I don't understand how importing works in Python:

1st specific situation:

When I import the same module in two different Python scripts, the module isn't imported twice, right? The first time Python encounters it, it is imported, and second time, does it check if the module has been imported, or does it make a copy?

2nd specific situation:

Consider the following module, called bla.py:

a = 10 

And then, we have foo.py, a module which imports bla.py:

from bla import *  def Stuff ():     return a 

And after that, we have a script called bar.py, which gets executed by the user:

from foo import * Stuff() #This should return 10  a = 5 Stuff() 

Here I don't know: Does Stuff() return 10 or 5?

like image 959
corazza Avatar asked May 08 '12 15:05

corazza


People also ask

What does Python import actually do?

In Python, you use the import keyword to make code in one module available in another. Imports in Python are important for structuring your code effectively. Using imports properly will make you more productive, allowing you to reuse code while keeping your projects maintainable.

How does Python import files?

To import a file in Python, use the import statement. The import statement combines two operations; it searches for the named module, then binds the results of that search to a name in the local scope.

Does import * import everything in Python?

Yes, it does. It imports everything (that is not a private variable, i.e.: variables whose names start with _ or __ ), and you should try not to use it according to "Properly importing modules in Python" to avoid polluting the local namespace. It is enough, but generally you should either do import project.

Does importing a Python script run it?

When you import a module in Python, all the code in it will be run, and all the variables in that module will be stuck on that module object.


1 Answers

Part 1

The module is only loaded once, so there is no performance loss by importing it again. If you actually wanted it to be loaded/parsed again, you'd have to reload() the module.

The first place checked is sys.modules, the cache of all modules that have been imported previously. [source]


Part 2

from foo import * imports a to the local scope. When assigning a value to a, it is replaced with the new value - but the original foo.a variable is not touched.

So unless you import foo and modify foo.a, both calls will return the same value.

For a mutable type such as a list or dict it would be different, modifying it would indeed affect the original variable - but assigning a new value to it would still not modify foo.whatever.

If you want some more detailed information, have a look at http://docs.python.org/reference/executionmodel.html:

The following constructs bind names: formal parameters to functions, import statements, class and function definitions (these bind the class or function name in the defining block), and targets that are identifiers if occurring in an assignment, for loop header, in the second position of an except clause header or after as in a with statement.

The two bold sections are the relevant ones for you: First the name a is bound to the value of foo.a during the import. Then, when doing a = 5, the name a is bound to 5. Since modifying a list/dict does not cause any binding, those operations would modify the original one (b and foo.b are bound to the same object on which you operate). Assigning a new object to b would be a binding operation again and thus separate b from foo.b.

It is also worth noting what exactly the import statement does:

  • import foo binds the module name to the module object in the current scope, so if you modify foo.whatever, you will work with the name in that module - any modifications/assignments will affect the variable in the module.
  • from foo import bar binds the given name(s) only (i.e. foo will remain unbound) to the element with the same name in foo - so operations on bar behave like explained earlier.
  • from foo import * behaves like the previous one, but it imports all global names which are not prefixed with an underscore. If the module defines __all__ only names inside this sequence are imported.

Part 3 (which doesn't even exist in your question :p)

The python documentation is extremely good and usually verbose - you find answer on almost every possible language-related question in there. Here are some useful links:

  • http://docs.python.org/reference/datamodel.html (classes, properties, magic methods, etc.) ()
  • http://docs.python.org/reference/executionmodel.html (how variables work in python)
  • http://docs.python.org/reference/expressions.html
  • http://docs.python.org/reference/simple_stmts.html (statements such as import, yield)
  • http://docs.python.org/reference/compound_stmts.html (block statements such as for, try, with)
like image 114
ThiefMaster Avatar answered Oct 09 '22 13:10

ThiefMaster