module
Understanding __name__
, __main__
, and Running Scripts as Modules in Python
1. The __name__
and __main__
Logic in Python
In Python, every module has a special built-in attribute called __name__
. This attribute is crucial for determining whether a module is being run directly or being imported into another module.
What is __name__
?
- When a module is run directly: If you run a Python script directly (e.g.,
python script.py
), Python assigns the string"__main__"
to__name__
in that script. - When a module is imported: If the module is imported into another script (e.g.,
import script
),__name__
is set to the module's name ("script"
in this case).
Using if __name__ == "__main__":
This conditional statement checks whether the module's __name__
is "__main__"
, which indicates the module is being run as the main program.
# example_module.py
def main():
print("Main function is running")
if __name__ == "__main__":
main()
- When run directly:
Output:
python example_module.py
Main function is running
- When imported into another module:
Output: *(No output, unless
# another_module.py
import example_module
main()
is explicitly called)*
Purpose of if __name__ == "__main__":
- Prevent Unintended Execution: Code inside this block only runs when the script is executed directly, not when it's imported. This is useful for test code or scripts that can be both imported as modules and run as standalone scripts.
- Organize Entry Point: It designates the starting point of your program when running the script.
2. Running a Script as a Module Using python -m
vs. Without -m
When you execute Python scripts, you have two options:
Option A: Running the Script Directly
python script.py
Characteristics:
- Execution Context: The script is executed as the
__main__
module. __name__
Value:__name__ == "__main__"
inscript.py
.- Imports:
- Absolute Imports: Work fine.
- Relative Imports: May fail because Python doesn't recognize
script.py
as part of a package.
Issue with Relative Imports:
If script.py
contains relative imports (e.g., from .module import something
), running it directly will cause an ImportError
:
ImportError: attempted relative import with no known parent package
Option B: Running the Script as a Module with python -m
python -m package.script
Characteristics:
- Execution Context: The script is run as part of the specified package.
__name__
Value:__name__ == "__main__"
inscript.py
.- Imports:
- Absolute Imports: Work fine.
- Relative Imports: Work because Python recognizes the package structure.
Benefits:
- Package Awareness: Python knows
script.py
is part ofpackage
, so relative imports resolve correctly. - Consistent Behavior: Aligns with how modules are imported elsewhere in your code.
Difference Between Running with and without -m
Running Without -m
(Direct Execution)
-
Imports:
- Absolute Imports: Need to reference the full path from the project root.
- Relative Imports: Fail because there's no package context.
-
Example:
python package/script.py
Potential Error:
ImportError: attempted relative import with no known parent package
Running With -m
(Module Execution)
-
Imports:
- Absolute Imports: Work correctly.
- Relative Imports: Work because Python recognizes the package hierarchy.
-
Example:
python -m package.script
The script runs successfully, and both absolute and relative imports work as expected.
Practical Example
Project Structure:
project/