ZetCode

Python Relative Imports

Last modified March 25, 2025

Relative imports in Python enable importing modules based on their position relative to the current module within a package hierarchy. They utilize dot notation to specify this relative path and are widely employed in Python packages. This tutorial examines their syntax, usage, and practical applications through detailed examples.

Relative imports are especially valuable in package development, enhancing maintainability and reducing fragility during reorganization. Unlike absolute imports, which define the full path from the project root, relative imports specify paths relative to the current module’s location. They follow the from .module import name syntax, where dots indicate the relative position within the package structure.

Basic Relative Import Syntax

This example illustrates the fundamental syntax of relative imports in Python.

package_structure
mypackage/
├── __init__.py
├── module1.py
└── subpackage/
    ├── __init__.py
    └── module2.py

This directory layout represents a simple package named mypackage. It contains a top-level module, module1.py, and a subpackage, subpackage, with its own module2.py. The __init__.py files mark these directories as Python packages.

subpackage/module2.py
from ..module1 import function1

def function2():
    return function1() + " from module2"

In module2.py, the statement from ..module1 import function1 performs a relative import. The two dots (..) instruct Python to ascend one package level to the parent directory, accessing module1.py. The function function2 then builds on function1, appending a string to its result.

The from ..module1 import function1 statement exemplifies a relative import. The double dots signify moving up one package level to import from module1.py in the parent package. Relative imports require a package context, enforced by __init__.py files, and cannot be used in standalone scripts.

Importing from Parent Package

This example demonstrates importing from a parent package using relative imports.

package_structure
mypackage/
├── __init__.py
├── utils.py
└── subpackage/
    ├── __init__.py
    └── module.py

This structure defines a package, mypackage, with a utility module, utils.py, at the top level and a subpackage, subpackage, containing module.py. The __init__.py files establish the package hierarchy.

subpackage/module.py
from ..utils import helper_function

def use_helper():
    return helper_function()

In module.py, the line from ..utils import helper_function imports helper_function from utils.py in the parent package. The two dots direct Python to move up one level, and use_helper invokes the imported function.

The from ..utils import helper_function statement retrieves helper_function from utils.py in the parent package. The double dots indicate an ascent of one level in the hierarchy. This method ensures imports remain concise and their origins clear relative to the current module.

Importing from Sibling Modules

This example shows how to import from a sibling module within the same package.

package_structure
mypackage/
├── __init__.py
├── module_a.py
└── module_b.py

This layout depicts a package, mypackage, with two sibling modules, module_a.py and module_b.py, at the same level. The __init__.py file designates mypackage as a package.

module_b.py
from .module_a import ClassA

class ClassB:
    def __init__(self):
        self.a = ClassA()

In module_b.py, the statement from .module_a import ClassA imports ClassA from the sibling module module_a.py. The single dot (.) refers to the current package, and ClassB instantiates ClassA in its constructor.

The from .module_a import ClassA statement accesses ClassA from module_a.py, a sibling in the same package. The single dot denotes the current package directory. This technique is ideal for organizing related functionality across modules at the same level.

Multi-level Relative Imports

This example illustrates relative imports across multiple package levels.

package_structure
mypackage/
├── __init__.py
├── module1.py
└── subpackage/
    ├── __init__.py
    ├── module2.py
    └── nested/
        ├── __init__.py
        └── module3.py

This structure outlines a package, mypackage, with a top-level module1.py and a subpackage, subpackage. Within subpackage, there’s module2.py and a nested subpackage, nested, containing module3.py. Each level has an __init__.py file.

nested/module3.py
from ...module1 import function1
from ..module2 import function2

def combined_function():
    return function1() + function2()

In module3.py, the line from ...module1 import function1 uses three dots to ascend two levels to import function1 from module1.py. The statement from ..module2 import function2 moves up one level to import function2 from module2.py. The combined_function merges their outputs.

The from ...module1 import function1 statement navigates two package levels up to reach module1.py, while from ..module2 import function2 ascends one level to module2.py. This shows how relative imports handle complex hierarchies with precision.

Relative Imports in __init__.py

This example demonstrates the use of relative imports in package initialization files to shape the package’s public API.

package_structure
mypackage/
├── __init__.py
├── core.py
└── utils/
    ├── __init__.py
    └── helpers.py

This hierarchy defines mypackage with a top-level core.py and a subpackage, utils, containing helpers.py. The __init__.py files establish the package and subpackage structure.

utils/__init__.py
from .helpers import utility_function
from ..core import CoreClass

__all__ = ['utility_function', 'CoreClass']

In utils/__init__.py, the line from .helpers import utility_function imports utility_function from helpers.py within the same subpackage. The statement from ..core import CoreClass reaches up to the parent package for CoreClass from core.py. The __all__ list defines the subpackage’s public interface.

The __init__.py file employs relative imports to expose selected items at the package level. Importing from .helpers targets a sibling module, while ..core accesses the parent package. This pattern simplifies access to key components in a package’s API.

Common Errors and Solutions

Best Practices for Relative Imports

Source

Python Relative Imports Documentation

In this article, we have explored Python relative imports and demonstrated their usage in package development through practical examples.

Author

My name is Jan Bodnar, and I am a passionate programmer with extensive programming experience. I have been writing programming articles since 2007. To date, I have authored over 1,400 articles and 8 e-books. I possess more than ten years of experience in teaching programming.

List all Python tutorials.