Python 元类 metaclass 详解

简介: - 元类是类的类,用于控制类实例的创建过程和初始化过程。- `__new__` 用于创建类实例,`__init__` 用于初始化实例。- 元类的主要作用是允许在类被创建之前修改它,如添加属性、修改行为。- 元类适用于对类创建过程进行深度干预的特殊需求,对于大多数情况,普通类定义和继承已经足够,元类提供额外的灵活性。

元类(metaclass)是 Python 中一个高级且相对较少使用的概念。元类可以被视为类的类,它控制类的创建过程。

一、基本概念

在 Python 中,一切皆对象。为了避免混淆,我们约定两个术语:

  • 类实例:当我们定义一个类时,实际上是在创建一个对象,这个对象就是类本身,我们将类本身这个对象称为 类实例

  • 类的实例:我们将使用类创建的实例称之为 类的实例

生活化的例子:

  • 我们可以将 元类 理解为 制作蛋糕模具的机床,将 理解为 制作蛋糕的模具
  • 制作蛋糕模具是元类创建的 类实例蛋糕 是蛋糕模具创建的 蛋糕类的实例

二、基本原理

元类 控制 类实例 的创建过程,其中 __new__ 方法用于创建 类实例,而 __init__ 方法用于初始化这个 类实例

控制 类的实例 的创建过程,其中 __new__ 方法用于创建 类的实例,而__init__方法用于初始化这个 类的实例

三、核心作用

元类的核心作用是控制类的创建过程,允许我们在类被实际创建之前修改它,这使得我们可以在类级别上进行高级的定制,例如添加额外的属性、修改类的行为等。

四、Demo 示例

示例 1:简单的元类

class SimpleMeta(type):
    registered_classes = []

    def __new__(cls, name, bases, dct):
        """创建类实例"""
        print("Generate new instance")
        return super().__new__(cls, name, bases, dct)

    def __init__(cls, name, bases, dct):
        """初始化类实例"""
        print("Init new instance")
        super().__init__(name, bases, dct)

        SimpleMeta.registered_classes.append(cls)


class MyClass1(metaclass=SimpleMeta):
    pass


print('========================================')


class MyClass2(metaclass=SimpleMeta):
    pass


print('========================================')

print(f"Registered classes: {SimpleMeta.registered_classes}")

输出结果:

Generate new instance
Init new instance
========================================
Generate new instance
Init new instance
========================================
Registered classes: [<class '__main__.SimpleMeta'>, <class '__main__.SimpleMeta'>]

示例说明:

元类 SimpleMeta 中的 __init__ 方法在 __new__ 方法执行后被调用,用于收集已注册的类的信息,展示了元类的实际用途。

示例 2:蛋糕模具定制

class ShapeMeta(type):
    def __new__(cls, name, bases, dct):
        # 获取类的名称并根据不同的类名进行定制
        if name == 'RoundCake':
            # 定制化操作
            dct['shape'] = 'Round'
        if name == 'TriangleCake':
            # 定制化操作
            dct['shape'] = 'Triangle'
        return super().__new__(cls, name, bases, dct)


class RoundCake(metaclass=ShapeMeta):
    pass


class TriangleCake(metaclass=ShapeMeta):
    pass


my_round_cake = RoundCake()
my_triangle_cake = TriangleCake()

print(my_round_cake.shape)  # 输出 Round
print(my_triangle_cake.shape)  # 输出 Triangle

输出结果:

Round
Triangle

示例说明:

元类 ShapeMeta 根据类的名称控制类的形状属性,这里可以看出元类对类创建过程的控制。

综合示例:面向切面编程

import functools


def listening(func):
    @functools.wraps(func)
    def wrapper(self, *args, **kwargs):
        print(f'do something before calling function {func.__name__}')
        result = func(self, *args, **kwargs)
        print(f'do something after calling function {func.__name__}')

    return wrapper


class ListeningMeta(type):

    def __new__(cls, name, bases, dct):
        for key, value in dct.items():
            if callable(value) and not key.startswith("__"):
                dct[key] = listening(value)
        return super().__new__(cls, name, bases, dct)


class MyClass(metaclass=ListeningMeta):
    def __init__(self, data):
        self._data = data

    @property
    def data(self):
        print(self._data)
        return self._data

    def do_somthing(self):
        print(f'do_somthing with {self._data}')


a = MyClass(1)
a.data
a.do_somthing()

输出结果:

1
do something before calling function do_somthing
do_somthing with 1
do something after calling function do_somthing

示例说明:

元类 ListeningMeta 展示了元类如何用于在不修改原始类代码的情况下添加额外的行为,实现面向切面编程的效果。

五、要点小结

  • 元类是类的类,用于控制类实例的创建过程和初始化过程。
  • __new__ 用于创建类实例,__init__ 用于初始化实例。
  • 元类的主要作用是允许在类被创建之前修改它,如添加属性、修改行为。
  • 元类适用于对类创建过程进行深度干预的特殊需求,对于大多数情况,普通类定义和继承已经足够,元类提供额外的灵活性。
目录
相关文章
|
4月前
|
开发者 Python
Python元类实战:打造你的专属编程魔法,让代码随心所欲变化
【7月更文挑战第7天】Python的元类是编程的变形师,用于创建类的“类”,赋予代码在构建时的变形能力。
56 1
|
4月前
|
设计模式 存储 Python
Python元类大揭秘:从理解到应用,一步步构建你的编程帝国
【7月更文挑战第6天】Python元类是创建类的对象的基石,允许控制类的生成过程。通过自定义元类,可在类定义时动态添加方法或改变行为。
75 0
|
22天前
|
开发者 Python
Python中的元类深度剖析与实战应用
Python中的元类深度剖析与实战应用
20 0
|
3月前
|
关系型数据库 MySQL Java
Python中的元类(metaclass)
Python中的元类(metaclass)
|
3月前
|
设计模式 存储 数据库连接
Python设计模式:巧用元类创建单例模式!
Python设计模式:巧用元类创建单例模式!
47 0
|
4月前
|
开发者 Python
元类,Python中的隐藏BOSS?掌握它,让你的编程之路畅通无阻
【7月更文挑战第7天】Python的元类是创建类的类,如同编程的“大BOSS”。它们让开发者在类创建时干预过程,添加功能,如自动注册、修改属性。元类通过`__new__`方法动态创建类,如示例中MetaClass得到Meta元类附加的属性。虽然使用需谨慎,以免增加复杂性,但元类提供了超越常规类的强大力量,解锁高级编程技术。
34 2
|
4月前
|
Python
告别平庸!Python元类:让你的代码设计思想跃升至新高度
【7月更文挑战第7天】Python的元类是创建类的类,用于在定义时定制类的行为。通过元类,可以在类创建时注入逻辑,比如添加方法或改变属性。例如,一个元类可以自动为所有类添加日志功能。元类应用广泛,如自动注册类、修改属性、实现单例模式和动态添加方法,提升代码设计的灵活性和效率。掌握元类,让代码设计超越常规。
28 1
|
4月前
|
Python
从入门到精通,Python元类:让你的代码拥有自我进化的超能力
【7月更文挑战第7天】元类是Python中类的类,用于定义类的创建和行为。它们允许在创建类时动态修改,常用于注册、属性修改或实现单例。要使用元类,需定义继承`type`的类,重写`__new__`或`__init__`。例如,一个简单的元类能自动给新类添加属性。虽然初学者会感到挑战,但通过实践能揭示其潜力,赋予代码超凡的灵活性。**
42 0
|
4月前
|
设计模式 开发者 Python
惊呆了!Python元类竟能如此玩转,你的编程世界将不再有界限
【7月更文挑战第6天】Python元类是类的类,用于控制类的创建。它们让开发者能自定义类的行为,如添加方法、改变继承或实例化过程。例如,定义一个元类`my_metaclass`,它会在创建类时自动添加新方法。元类广泛应用在单例、插件系统和ORM等高级场景,拓展了Python代码的灵活性和威力。掌握元类,揭开编程魔法的面纱,为代码解锁更多可能。
23 0
|
4月前
|
设计模式 Python
深度揭秘!Python元类:掌握它,让你的代码拥有创造类的能力
【7月更文挑战第6天】Python元类探秘:**元类是类的类,用于控制类的创建。通过定义元类,可自定义类的行为,如动态添加方法或改变继承结构。示例中,`my_metaclass`在创建类时添加`new_method`。元类强大且适用于高级编程,如动态修改、注册类或实现设计模式。理解并善用元类能提升Python编程技巧。
42 0