How to Execute a Python Script#

There are two primary ways to execute a Python script.

  1. You can pass your script to Python in your shell

  2. You can call your script directly as a command in your shell

1. Pass your script to the Python command#

The first way to execute a Python script is to use the python command. The python command takes the name of a file and executes it from your shell, like this:

python my_program.py

When Python reads a file using this approach, it executes each line of code from top to bottom. Lines within functions or classes are not executed, unless that function is called or class is created outside of any other function or class definition. This is similar, but not identical, to the behavior of copying this file and pasting it line-by-line into an interactive Python shell (or a Jupyter notebook cell).

def report_error():
  print("An error has occured")

print("\N{Sparkles} Hello from Python \N{Sparkles}")

Note that only one line is printed when this script is run.

python my_program.py
# ✨ Hello from Python ✨

The report_error function only runs if you call it directly in the file:

def report_error():
  print("An error has occured")

print("\N{Sparkles} Hello from Python \N{Sparkles}")
report_error()
python my_program.py
# ✨ Hello from Python ✨
# An error has occured

2. Associate the script with a launch command#

The other way a Python script may be executed is to associate the file with a launch command. The way in which this association is done depends on what operating system you are running.

Executing Python scripts on macOS / Linux / Non-Windows#

On Linux or Mac systems, the Python file can itself be turned into a command. By adding a shebang as the first line in any Python file, and by giving the file executable permissions the file can be directly invoked without a python command.

#!/usr/bin/env python
# The above line is a shebang, and can take the place of typing python on the command line
# This comment is below, because shebangs must be the first line!

def shiny_hello():
    print("\N{Sparkles} Hello from Python \N{Sparkles}")

shiny_hello()
my_program.py
# ✨ Hello from Python ✨

Tip

Shebangs are a feature of POSIX. POSIX represents some level of compatibility between systems. Linux, macOS, all BSDs, and many other operating systems are fully- or mostly-POSIX compliant.

Windows is not natively POSIX compliant. However, some “modes” inside of Windows are, such as WSL (Windows Subsystem for Linux), gitbash, or some VSCode terminals.

Executing Python scripts on Windows#

If your Windows machine has Python registered as the default application associated with .py files, then any Python scripts can be run as commands. However, only one Python can be registered at a time, so all Python scripts run this way will use the same Python environment.

Additionally, most Windows Python installs come with the Python Launcher. This launcher allows you to specify the version of Python to run and also can read shebang lines and emulate some of that behavior. This allows you to reuse shebang lines between Linux, macOS, and Windows systems. However, on Windows, the command must still be prefaced with another command (py).

py my_program.py
# ✨ Hello from Python ✨

Tip

All Python files should end in a .py. The .py is required for Windows to associate a script with Python. In comparison, on Linux and Mac machines, .py is a recommended best practice; the shebang (#!) associates the script execution with Python.

Windows doesn’t rely on a shebang (#!) to determine how to execute a Python file. Instead, Python sees the first # and treats the line as a comment and ignores it!

For reproducibility, which ensures that your code runs consistently across machines, you should always include the .py extension and the shebang (#!) in your Python scripts.

Comparing passing files to Python vs the “shebang” (#!) execution approach#

There are pros and cons of using the two approaches above to execute Python files.

Pros of passing a file to python:#

  • It works with all files: The file path passed to Python doesn’t have to end in a .py (not even on Windows)

  • It works with more restrictive fiels: The file doesn’t need execute permissions

  • It’s reproducible: This approach works on every system

  • You are being explicit about what you expect to happen

Pros of associating your script with a launcher:#

  • The Python version used is associated with a specific Python installation

  • That Python can be different that the Python installation currently active in a virtual-environment

  • You don’t have to remember when using the command which Python installation the script should be associated with, or even that the script is written in Python

Separating script from import behavior#

Sometimes a Python file that is useful to execute is also useful to import. You may want to use shiny_hello in another Python file. But right now, the my_program.py does all its script behavior even when it is imported. Consider

import my_program

def guess_my_number():
    my_program.shiny_hello()
    print("Was your number 42?")

guess_my_number()
# ✨ Hello from Python ✨
# ✨ Hello from Python ✨
# Was your number 42?

You may have expected it to print hello only once, but it printed the same line twice. This is because my_program is written to always call shiny_hello. Now guess_my_number also calls shiny_hello. Remember the contents of my_program.py.

#!/usr/bin/env python
# The above line is a shebang, and can take the place of typing python on the command line
# This comment is below, because shebangs must be the first line!

def shiny_hello():
    print("\N{Sparkles} Hello from Python \N{Sparkles}")

# NOTE: my_program runs the shiny_hello function whenever it is loaded
shiny_hello()

How can you make my_program only call shiny_hello when it is used as a script? You may have already seen the answer, without realizing what it was doing. my_program needs a conditional that checks if is is in “script mode” or “import mode” and that conditional is if __name__ == "__main__":.

This conditional is most often included at the bottom of modules that are expected to be executed directly. This conditional separates code that is intended to execute as part of a command or script from code that is intended to execute as part of an import within another script.

#!/usr/bin/env python
# The above line is a shebang and can take the place of typing python on the command line
# This comment is below, because shebangs must be the first line!

def shiny_hello():
    print("\N{Sparkles} Hello from Python \N{Sparkles}")

# By calling shiny_hello within this conditional, it will only run when this file is run as a script. It will
# not run if you import it into another script.
if __name__ == "__main__":
    shiny_hello()
my_program.py
# ✨ Hello from Python ✨
import my_program

def guess_my_number():
    my_program.shiny_hello()
    print("Was your number 42?")

guess_my_number()
# ✨ Hello from Python ✨
# Was your number 42?

Note

Why did that work?

All Python modules (individual files) have a __name__ attribute, which is usually the same as the name used to import the module.

import os
print(os.__name__)
# 'os'

This attribute is available within a module by using the variable __name__. So in the os.py module, __name__ also gives the value 'os'.

Importantly, this name is changed for the first user-module executed by Python. When you pass a file to python, that is the first user-module executed. For this module, and only when it is the first, the __name__ is changed to the string '__main__'. This answers the question for every module used in a Python program, “am I the main module?”.