ZetCode

Python Metaclasses

Last modified March 26, 2025

This detailed guide explores Python metaclasses, which are classes of classes that govern the creation and behavior of other classes. Through practical examples, we'll examine their core concepts, showcasing how they unlock advanced metaprogramming capabilities in Python's object-oriented framework.

In Python, metaclasses are advanced tools that define the behavior and structure of classes. A metaclass controls how classes are created and can modify their attributes or methods dynamically. By specifying a metaclass using the metaclass keyword, developers can implement custom behavior, such as enforcing coding standards or adding additional logic to class creation. Essentially, metaclasses act as the "class of a class," providing greater flexibility and control in object-oriented programming.

Understanding the Type-Metaclass Relationship

To grasp metaclasses, it's essential to explore Python's type system and its foundational structure.

type_metaclass.py
class SimpleClass:
    pass

print(type(SimpleClass))  # <class 'type'>
print(type(type))        # <class 'type'>

instance = SimpleClass()
print(type(instance))    # <class '__main__.SimpleClass'>

In this example, the type function unveils Python's type hierarchy. For SimpleClass, it returns <class 'type'>, indicating that SimpleClass is an instance of the type metaclass. Similarly, type(type) yields <class 'type'>, showing that type is its own metaclass. For an instance, type(instance) identifies it as <class '__main__.SimpleClass'>.

This example reveals that regular classes like SimpleClass are instances of type, which itself is an instance of type, establishing it as the root metaclass. Instances of classes, however, belong to their specific class type. This hierarchy positions metaclasses at the apex of Python's type system, overseeing class construction before any instances are created, thus enabling deep customization of class behavior.

Creating a Basic Metaclass

Metaclasses are crafted by subclassing type, Python's default metaclass, to intercept and modify class creation.

basic_metaclass.py
class Meta(type):
    def __new__(cls, name, bases, namespace):
        print(f"Creating class {name}")
        return super().__new__(cls, name, bases, namespace)

class MyClass(metaclass=Meta):
    pass

# Output when script runs: "Creating class MyClass"

Here, Meta inherits from type, defining a metaclass. The __new__ method, invoked during class creation, logs the class name and delegates construction to super().__new__. When MyClass is defined with metaclass=Meta, it prints "Creating class MyClass" as the class is formed, demonstrating the metaclass's intervention.

By inheriting from type, Meta gains control over class creation. The __new__ method receives the metaclass itself as cls, the class name as name, a tuple of base classes as bases, and a dictionary of class attributes as namespace. Though simple, this metaclass illustrates the core mechanism, with actual creation handled by type's implementation via super().

Modifying Class Attributes

Metaclasses can inspect and alter class attributes during their creation, enforcing conventions or transformations.

attribute_modification.py
class UpperAttrMeta(type):

    def __new__(cls, name, bases, namespace):
        upper_namespace = {
            key.upper(): value 
            for key, value in namespace.items()
            if not key.startswith('__')
        }

        return super().__new__(cls, name, bases, upper_namespace)

class Demo(metaclass=UpperAttrMeta):
    x = 10
    y = 20

print(Demo.X)  # 10
print(Demo.Y)  # 20
# print(Demo.x) would raise AttributeError

The UpperAttrMeta metaclass transforms attribute names to uppercase in Demo. It constructs a new upper_namespace dictionary, converting keys like x to X while preserving values, excluding dunder methods (e.g., __init__). The modified namespace is passed to super().__new__, so Demo.X accesses 10, but Demo.x raises an AttributeError.

This metaclass creates a new namespace with uppercase keys from the original, skipping special methods to avoid breaking Python's internals. The altered namespace is then used to construct the class. Such transformations are valuable for enforcing naming standards or adapting attributes for frameworks, demonstrating the metaclass's power to reshape class definitions dynamically.

Singleton Pattern with Metaclass

Metaclasses offer an elegant way to implement the Singleton pattern, ensuring only one instance of a class exists.

singleton_metaclass.py
class SingletonMeta(type):
    _instances = {}
    
    def __call__(cls, *args, **kwargs):
        if cls not in cls._instances:
            cls._instances[cls] = super().__call__(*args, **kwargs)
        return cls._instances[cls]

class Database(metaclass=SingletonMeta):
    def __init__(self):
        print("Initializing database connection")

db1 = Database()
db2 = Database()
print(db1 is db2)  # True

In SingletonMeta, the __call__ method overrides instance creation. It maintains a class-level _instances dictionary, creating a new instance via super().__call__ only if none exists for cls. For Database, db1 triggers initialization, but db2 reuses it, confirming identity with True.

This approach overrides __call__ to manage instantiation, storing singletons in a dictionary and returning existing instances when available. Compared to decorators, this method is inherited by subclasses, harder to bypass, and centralizes logic, making it a robust choice for ensuring a single instance, such as a database connection, across an application.

Class Registration Pattern

Metaclasses can automatically register classes in a central registry, ideal for plugin or extension systems.

class_registration.py
class PluginMeta(type):
    registry = {}
    
    def __new__(cls, name, bases, namespace):
        new_class = super().__new__(cls, name, bases, namespace)
        if not name.startswith('Base'):
            cls.registry[name.lower()] = new_class
        return new_class

class BasePlugin(metaclass=PluginMeta):
    pass

class DataPlugin(BasePlugin):
    pass

class AuthPlugin(BasePlugin):
    pass

print(PluginMeta.registry)  
# {'dataplugin': <class '__main__.DataPlugin'>, 
#  'authplugin': <class '__main__.AuthPlugin'>}

PluginMeta registers subclasses of BasePlugin in its registry. After creating new_class with super().__new__, it adds concrete classes (excluding those named like "Base") to the dictionary with lowercase keys. The output shows DataPlugin and AuthPlugin automatically registered.

This pattern tracks all subclasses of BasePlugin without extra code, making the registry accessible via the metaclass. By filtering out base classes, it ensures only concrete implementations are logged. This is particularly powerful for plugin systems, enabling dynamic discovery and management of extensions in a clean, automated manner.

Interface Enforcement

Metaclasses can enforce that subclasses implement specific methods, acting as a runtime contract checker.

interface_enforcement.py
class InterfaceMeta(type):
    required_methods = ['save', 'load']
    
    def __new__(cls, name, bases, namespace):
        if not any('__module__' in ns for ns in namespace.values()):
            for method in cls.required_methods:
                if method not in namespace:
                    raise TypeError(f"Missing required method: {method}")
        return super().__new__(cls, name, bases, namespace)

class Storage(metaclass=InterfaceMeta):
    pass

class DatabaseStorage(Storage):
    def save(self, data):
        pass
    
    def load(self, id):
        pass

# This would raise TypeError:
# class BadStorage(Storage): pass

InterfaceMeta defines save and load as mandatory in required_methods. During class creation, it checks the namespace for these methods, raising a TypeError if any are missing, unless the class is imported (detected via __module__). DatabaseStorage complies, while an uncommented BadStorage would fail.

This metaclass specifies required methods and verifies their presence, bypassing checks for imported classes to avoid false positives. It raises an exception if the contract is unmet, offering a flexible alternative to abstract base classes. This ensures interface compliance at class definition time, enhancing code reliability and maintainability.

Method Wrapping

Metaclasses can wrap class methods to inject additional behavior, such as logging or monitoring, transparently.

method_wrapping.py
class LoggedMeta(type):

    def __new__(cls, name, bases, namespace):
        for attr_name, attr_value in namespace.items():
            if callable(attr_value):
                namespace[attr_name] = cls.log_method(attr_value)
        return super().__new__(cls, name, bases, namespace)
    
    @staticmethod
    def log_method(method):
        def wrapped(*args, **kwargs):
            print(f"Calling {method.__name__}")
            return method(*args, **kwargs)
        return wrapped

class Service(metaclass=LoggedMeta):
    def process(self, data):
        return data.upper()

s = Service()
s.process("test")  # Prints "Calling process" then returns "TEST"

LoggedMeta scans the namespace for callable attributes, replacing each with a wrapped version via log_method. The wrapper logs the method name before invoking the original, as seen when s.process("test") outputs "Calling process" and "TEST". This enhances the Service class without altering its source.

This metaclass iterates over attributes, identifies methods, and substitutes them with wrappers that add logging functionality. The wrapped methods retain their original behavior while addressing cross-cutting concerns like logging, timing, or authorization. This approach is particularly useful for applying consistent enhancements across all methods of a class seamlessly.

Dynamic Attribute Creation

Metaclasses can dynamically generate attributes based on class definitions, streamlining and optimizing class construction.

dynamic_attributes.py
class AutoSlotsMeta(type):

    def __new__(cls, name, bases, namespace):
        if '__annotations__' in namespace:
            namespace['__slots__'] = tuple(namespace['__annotations__'].keys())
        return super().__new__(cls, name, bases, namespace)

class Person(metaclass=AutoSlotsMeta):

    name: str
    age: int
    
    def __init__(self, name, age):
        self.name = name
        self.age = age

p = Person("Alice", 30)
# p.address = "Street" would raise AttributeError

AutoSlotsMeta checks for __annotations__ in the namespace, converting its keys (name, age) into a __slots__ tuple. For Person, this restricts attributes to name and age, so p.address fails with an AttributeError, while initialization works as expected.

This metaclass leverages type annotations to define __slots__, enhancing memory efficiency by limiting instance attributes. Unlike manual __slots__, it avoids repetition, reduces boilerplate, and keeps annotations as the single source of truth. This pattern simplifies class design while optimizing resource use dynamically.

Class Versioning

Metaclasses can embed versioning metadata into classes, facilitating tracking and management over time.

class_versioning.py
import time

class VersionedMeta(type):
    def __new__(cls, name, bases, namespace):
        namespace['created_at'] = time.time()
        namespace['version'] = 1
        return super().__new__(cls, name, bases, namespace)

class Document(metaclass=VersionedMeta):
    pass

print(Document.created_at)  # Unix timestamp
print(Document.version)     # 1

VersionedMeta adds created_at (a timestamp) and version (set to 1) to the namespace of Document. When accessed, Document.created_at yields the creation time in seconds since the epoch, and Document.version shows the initial version number, providing metadata about the class.

This versioning system automatically attaches metadata, such as creation timestamps and version numbers, to classes. It enables runtime inspection and could be extended to increment versions, generate changelogs, or enforce compatibility. Such capabilities are valuable for auditing, debugging, or managing evolving class definitions in large systems.

Multiple Inheritance with Metaclasses

Python manages metaclass inheritance systematically, allowing multiple metaclasses to collaborate in class creation.

multiple_inheritance.py
class MetaA(type):
    def __new__(cls, name, bases, namespace):
        namespace['a'] = 1
        return super().__new__(cls, name, bases, namespace)

class MetaB(type):
    def __new__(cls, name, bases, namespace):
        namespace['b'] = 2
        return super().__new__(cls, name, bases, namespace)

class CombinedMeta(MetaA, MetaB):
    pass

class MyClass(metaclass=CombinedMeta):
    pass

print(MyClass.a)  # 1
print(MyClass.b)  # 2

MetaA and MetaB each add an attribute (a and b) to the namespace. CombinedMeta inherits from both, and MyClass uses it as its metaclass. The resulting class inherits both attributes, with MyClass.a yielding 1 and MyClass.b yielding 2, showing combined effects.

When using multiple metaclasses, Python ensures compatibility, allowing combination through inheritance. The most derived metaclass, CombinedMeta, governs creation, with each parent's __new__ contributing attributes. This demonstrates how metaclasses can collaborate, providing a flexible way to compose class behaviors from multiple sources.

Metaclass for Attribute Validation

Metaclasses can validate attribute values or types during class creation, ensuring correctness before instantiation.

attribute_validation.py
class ValidateMeta(type):

    def __new__(cls, name, bases, namespace):
        for attr, value in namespace.items():
            if attr == 'max_size' and not isinstance(value, int):
                raise ValueError(f"'max_size' must be an integer, got {type(value)}")
        return super().__new__(cls, name, bases, namespace)

class Buffer(metaclass=ValidateMeta):
    max_size = 1024

# This would raise ValueError:
# class BadBuffer(metaclass=ValidateMeta):
#     max_size = "large"

ValidateMeta checks the namespace for a max_size attribute, ensuring it's an integer. For Buffer, max_size = 1024 passes, but an uncommented BadBuffer with max_size = "large" would trigger a ValueError, halting class creation with a type mismatch error.

This metaclass inspects attributes like max_size, enforcing type constraints at class definition time. By raising exceptions for invalid values, it prevents runtime errors, offering a proactive way to validate class configuration. Such validation is crucial for settings or constants that must meet specific criteria in a system.

Metaclass with Custom Initialization

Metaclasses can customize class initialization, adding behavior when classes are first defined.

custom_init_metaclass.py
class InitMeta(type):

    def __init__(cls, name, bases, namespace):
        super().__init__(name, bases, namespace)
        print(f"Class {name} initialized with {len(bases)} base classes")

class Base:
    pass

class Derived(Base, metaclass=InitMeta):
    pass

# Output: "Class Derived initialized with 1 base classes"

InitMeta overrides __init__, called after class creation, to log the class name and number of base classes. When Derived is defined with Base as its parent, it prints "Class Derived initialized with 1 base classes", reflecting its inheritance structure.

This metaclass enhances class initialization by executing custom logic post- creation, leveraging __init__. It accesses the class's name and bases, providing insight into its structure immediately after definition. This is useful for logging, setup tasks, or triggering initialization hooks in complex class hierarchies.

Metaclass for Method Overriding

Metaclasses can override or extend existing methods, modifying behavior without altering the original class code.

method_override.py
class OverrideMeta(type):
    def __new__(cls, name, bases, namespace):
        if 'compute' in namespace:
            original = namespace['compute']
            def enhanced_compute(self, x):
                return original(self, x) * 2
            namespace['compute'] = enhanced_compute
        return super().__new__(cls, name, bases, namespace)

class Calculator(metaclass=OverrideMeta):
    def compute(self, x):
        return x + 1

calc = Calculator()
print(calc.compute(5))  # 12 (instead of 6)

OverrideMeta checks for a compute method in the namespace, replacing it with enhanced_compute that doubles the original result. For Calculator, compute(5) originally returns 6 (5 + 1), but the metaclass adjusts it to 12 (6 * 2), demonstrating the override.

This metaclass detects and modifies specific methods like compute, preserving the original while extending its functionality. It's a powerful technique for enhancing behavior across classes, such as amplifying results or adding preprocessing, without requiring direct changes to the class definition, thus maintaining flexibility and reusability.

Best Practices and Warnings

Source

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.