42、Python之面向对象:元类应用于定义检查、动态注入、插件注册

news2025/1/16 15:42:46

引言

在上一篇文章中,我们简单聊了一下Python中元类的概念,以及如何定义一个简单的元类。虽然,我们已经学会了定义元类,但是,相信不少刚接触Python的同学,大多会产生这样的疑问:这个不太好理解的元类,费工夫学来,到底有什么用呢?

这篇文章中,我们就接着来介绍下Python中元类的应用场景。

当然,需要提前说明的是,元类的使用,属于相对高级一些的用法,正常业务场景中一般没有必要用到元类。但是,了解了元类的定义极其适用场景,在后续阅读框架源码或者进行框架功能的开发时,还是很有必要的。

类定义检查

由于元类能够控制类对象的创建,所以,如果需要对类的定义进行检查、校验,比如必须写方法的文档注释,必须实现特定方法等,则可以在元类中来实现这些逻辑。

比如,假如有这样一个场景,要用到我们的框架的,在用户自定义类时,必须基于框架提供的元类进行类的定义,同时方法的定义中必须添加文档描述。

可以通过如下代码进行模拟:

class Model(type):
    def __new__(cls, name, bases, namespace, **kwargs):
        for key, value in namespace.items():
            if key.startswith('__'):
                continue
            if not hasattr(value, '__call__'):
                continue
            if not getattr(value, '__doc__'):
                raise TypeError(f"方法{key}必须添加docstring描述")
        return super().__new__(cls, name, bases, namespace, **kwargs)


class DaGongRen(metaclass=Model):
    def __init__(self, name):
        self.name = name

    def save(self):
        """ 这是方法描述 """
        print(f"save()方法,可以是保存到数据库的逻辑")

    def work(self):
        print(f"{self.name}在努力工作")


if __name__ == '__main__':
    pass

执行结果:

d8c44cb8833ff0a6b1d7efeb3e59681a.jpeg

代码中,我们定义了一个元类Model,在基于Model进行类的定义(也就是创建类对象)时,会调用__new__()方法。所以,我们将类定义的校验逻辑,添加到了该方法中。简单介绍一下校验逻辑:

1、以__打头的属性、方法,统一不做校验。

2、没有__call__属性的,粗略来说,就是不可调用的对象也不做检查。

3、经过1、2的逻辑,我们只会对方法进行检查,而如果一个方法对象没有__doc__属性的话,说明没有写文档字符串,则抛出TypeError的异常。

需要补充说明的是,当我们在元类的__new__方法中添加校验逻辑时,可以阻断类对象的创建,也就是类定义本身就会执行终止。如果我们只是需要添加警告提醒、不阻断类对象的创建,也可以将这些逻辑移到__init__()方法中。

动态注入

既然,我们可以通过元类进行类定义的检查,那么我们同样可以在类对象的真实创建过程进行一些额外的加工,比如统一添加类属性,统一添加方法等。这样来说,我们在代码复用的实现上,除了继承外,又多了一条途径。

比如,我们想通过元类给每个类都添加一个方法:inject_method

class Model(type):
    def __new__(cls, name, bases, namespace, **kwargs):
        namespace['inject_method'] = lambda self: f'这是一个动态添加到{self.__class__}中的方法'
        return super().__new__(cls, name, bases, namespace, **kwargs)


class DaGongRen(metaclass=Model):
    def __init__(self, name):
        self.name = name


class XueShengDang(metaclass=Model):
    def __init__(self, name):
        self.name = name


if __name__ == '__main__':
    zs = DaGongRen('张三')
    ls = XueShengDang('李四')
    print(zs.inject_method())
    print(ls.inject_method())

执行结果:

4d81bb8ec4274b6d410efc2832839ec0.jpeg

在代码中,我们定义了一个元类Model,并通过该元类定义了两个类,没有通过继承关系,我们通过元类的__new__()方法,给这两个类动态添加了一个inject_method()方法。

插件注册

有一种场景是,在一个基于插件化设计的系统中,我们需要能够动态地加载和使用插件,从而提供动态的系统扩展功能。这种场景需求,我们也可以通过元类来实现。

我们可以通过在系统中维护一个全局的插件类注册表,每次通过元类进行插件类的扩展时,都可以在全局的注册表中进行注册。

以代码来模拟这种场景:

plugin_registry = {}


class Plugin(type):
    def __new__(cls, name, bases, namespace, **kwargs):
        if 'execute' not in namespace or not hasattr(namespace['execute'], '__call__'):
            raise TypeError(f'{name}类未定义execute()方法')
        new_class = super().__new__(cls, name, bases, namespace, **kwargs)
        plugin_registry[name] = new_class
        return new_class


class PluginA(metaclass=Plugin):
    def execute(self):
        print(f"插件{self.__class__()}在运行")


class PluginB(metaclass=Plugin):
    def execute(self):
        print(f"插件{self.__class__()}在运行")


if __name__ == '__main__':
    for key, value in plugin_registry.items():
        value().execute()


执行结果:

a190a391858a5391aab58933438026b5.jpeg

总结

本文简单介绍了Python中元类在类定义检查、动态属性/方法注入、插件注册等场景中的使用。当然,这些功能在通常的业务场景中是用不到的。但是,如果有同学要阅读框架源码,或者打算自己开发框架,相信元类的知识一定会对你有所帮助。

感谢您的拨冗阅读,如果对您学习Python有所帮助,欢迎点赞、关注。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2052479.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Leetcode 1189 】 “气球” 的最大数量 —— 数组模拟哈希表

给你一个字符串 text,你需要使用 text 中的字母来拼凑尽可能多的单词 "balloon"(气球)。 字符串 text 中的每个字母最多只能被使用一次。请你返回最多可以拼凑出多少个单词 "balloon"。 示例 1: 输入&#…

YOLOv5改进 | 融合改进 | C3融合重写星辰网络之Rewrite the Stars⭐【CVPR2024】

秋招面试专栏推荐 :深度学习算法工程师面试问题总结【百面算法工程师】——点击即可跳转 💡💡💡本专栏所有程序均经过测试,可成功执行💡💡💡 专栏目录: 《YOLOv5入门 改…

Positional Encoding | 位置编码【详解】

文章目录 1、位置编码的2种方案2、位置编码3、公式详解 : 绝对位置 、 相对位置4、代码4.1 代码14.2 代码2 1、位置编码的2种方案 transformer的作者刚开始说固定的位置编码和可学习的位置编码的效果是差不多的,后来证明可学习的位置编码没有太大的必要&…

聊一下订单超时自动关闭的几种方式以及使用场景

订单超时自动关闭的本质其实是一种延时的功能实现,具体实现方式有很多种,但是我们方式的选择是需要结合业务场景的,没有更好的方案,只有更适合的方案,所以我们必须要结合自己的实际业务,以及业务的后续发展…

政务大数据解决方案(六)

政务大数据解决方案通过构建全面的数据集成平台,将来自不同政府部门和公共服务领域的多维度数据汇聚起来,运用先进的数据分析和人工智能技术进行深度挖掘与预测,从而为政府提供实时、精准的信息支持,优化决策流程,提高…

工业4.0下的防勒索病毒策略:主机加固在工控机防病毒中前行

MCK主机加固产品是一款专注于数据安全解决方案的软件,它在防勒索病毒和工控机防病毒方面具备一系列独特的功能和优势。 防勒索病毒方面: 内核级签名校验技术:MCK系统通过这项技术对操作系统启动及加载的所有模块进行可信认证,确…

电脑只有一个盘怎么分区?单盘电脑的解决方案

在现代计算机使用中,硬盘分区是一个重要的环节,它有助于我们更好地管理数据、提高系统运行效率,并在一定程度上保护数据的安全。 然而,当我们购买了一台新电脑后发现电脑里只有一个硬盘分区,这时候我们可能就需要对电…

Golang | Leetcode Golang题解之第349题两个数组的交集

题目&#xff1a; 题解&#xff1a; func intersection(nums1 []int, nums2 []int) (res []int) {sort.Ints(nums1)sort.Ints(nums2)for i, j : 0, 0; i < len(nums1) && j < len(nums2); {x, y : nums1[i], nums2[j]if x y {if res nil || x > res[len(re…

什么是CDN,CDN的作用是什么?

CDN CDN英文全称Content Delivery Network&#xff0c;即为内容分发网络。它是建立并覆盖在承载网之上&#xff0c;将内容存储在分布式的服务器上&#xff0c;使⽤户可以从距离较近的服务器获取所需的内容&#xff0c;从⽽减少数据传输的时间和距离&#xff0c;提⾼内容的传输…

基于Java的同城宠物服务预约系统的设计与实现---附源码78744

摘要 随着人们对宠物的喜爱和关注度不断增加&#xff0c;对宠物服务的需求也日益增长。为了提供更便捷、高效的宠物服务&#xff0c;设计并实现了一款基于 Java 的同城宠物服务预约系统。 本系统旨在满足宠物主人对各种宠物服务的需求&#xff0c;同时为宠物服务提供者提供一个…

C++ 设计模式——抽象工厂模式

抽象工厂模式 抽象工厂模式 抽象工厂模式主要组成部分代码实现抽象工厂模式模式的 UML 图抽象工厂模式 UML 图解析优点和缺点适用场景 抽象工厂模式提供一个接口&#xff0c;用于创建一系列相关或相互依赖的对象&#xff0c;而无需指定它们的具体类。它通常用于需要创建多个产品…

电脑怎么截图?截屏电脑快捷键ctrl加什么?

截图是我们日常使用电脑过程中非常常见的操作之一。无论是想保存有用的信息、分享有趣的内容&#xff0c;还是记录某个错误信息&#xff0c;截图都是一个简单而有效的方式。但是&#xff0c;不同的操作系统和需求会决定使用不同的方法来截图。接下来&#xff0c;我们将详细介绍…

opencascade Bnd_Range源码学习区间计算

opencascade Bnd_Range 前言 这个类描述了由两个实数值限定的 1D 空间中的区间。 一个区间可以是无效的&#xff0c;这表示区间中不包含任何点。 方法 1 默认构造函数。创建一个无效区间。 Bnd_Range() &#xff1b; 2 构造函数。创建最小最大值区间 Bnd_Range(const Sta…

使用LoRA对Llama3微调

使用LoRA&#xff08;Low-Rank Adaptation of Large Language Models&#xff09;技术对Llama-3语言模型进行微调。 理论知识参考百度安全验证 微调的前提条件 现在huggingface上下载llama2或llama3的huggingface版本。 我下载的是llama-2-13b-chat。 大语言模型微调方法 …

ComfyUI的部署,Ubuntu22.04系统下——点动科技

在服务器Ubuntu22.04系统下&#xff0c;ComfyUI的部署 一、ubuntu22.04基本环境配置1.1 更换清华Ubuntu镜像源1.2 更新包列表&#xff1a;2. 安装英伟达显卡驱动2.1 使用wget在命令行下载驱动包2.2 更新软件列表和安装必要软件、依赖2.2 卸载原有驱动2.3 安装驱动2.4 安装CUDA2…

【实战】分组校验

在实际的业务场景中同一个Entity的校验可能会有不同的规则&#xff0c;比如添加数据品牌id必须为空&#xff0c;而更新数据品牌Id必须不为空&#xff0c;针对这种情况我们需要使用分组校验来实现 在Entity中指定分组规则 使用 /*** 保存*/RequestMapping("/save")pub…

[STM32]如何正确的安装和配置keil?(详细)

一、我们为什么需要keil? 对于嵌入式开发的硬件来讲STM32可以说有着不可撼动的地位&#xff0c;它可能是很多人入门嵌入式开发接触到的第一款芯片&#xff0c;其强大的生态和大量开放的源代码也深受开发者的喜爱。对于嵌入式开发的软件来讲&#xff0c;keil绝对是在一届软件中…

知识竞赛中限时答题环节竞赛规则有哪些设计方案

限时答题在知识竞赛活动中是一个比较新颖的玩法&#xff0c;通过在一定时间内快速答题来提高现场紧张气氛&#xff0c;达到很好的现场效果。这种方式要求选手不但要答题正确&#xff0c;还要答题速度。那么&#xff0c;常用的限时答题环节规则应怎么设计呢&#xff1f;下面列出…

智能数字矿山钻机机械设备类网站模板

智能数字矿山钻机设备类网站模板&#xff0c;非常高端大气上档次&#xff01;易优内容管理系统是一套专注中小型企业信息传播解决方案的管理系统&#xff0c;更是一套后台管理框架&#xff0c;可以通过个性定制导航入口&#xff0c;扩展前端多个场景&#xff0c;比如可以用于小…

后端开发刷题 | 链表内指定区间反转【链表篇】

描述 将一个节点数为 size 链表 m 位置到 n 位置之间的区间反转&#xff0c;要求时间复杂度 O(n)O(n)&#xff0c;空间复杂度 O(1)O(1)。 例如&#xff1a; 给出的链表为 1→2→3→4→5→NULL1→2→3→4→5→NULL, m2,n4 返回 1→4→3→2→5→NULL 数据范围&#xff1a; 链表…