python高级3——元类与动态类创建

article/2025/9/7 11:47:00

元类

1、元类(Metaclass)的概念

  • 元类是Python中一个高级概念,本质上是**“类的类”**,用于控制创建类的过程。元类是创建类的模板
  • 在Python中:
    • 类定义了实例的行为
    • 元类定义了类的行为

2、python中的默认元类

  • 在Python中,type是所有类的默认元类。

  • 我们所熟知的object基类就是type元类的一个实例

  • 当使用class关键字创建类时,因为没有指定元类,因此python会寻找父类的元类,如果父类也没有元类的话会退回到最初的type类。

  • 找到元类后,基于元类创建类其实就是执行了元类中的__new____init__ 方法来创建一个类
    在这里插入图片描述

    class MyClass:pass
    # 等价于
    MyClass = type('MyClass', (), {})
    

    使用上述代码创建一个类,实际上是以下代码的语法糖:

    MyClass = type('MyClass', (), {})
    

3、创建元类的过程

  • 因为创建类其实就是执行元类中的__new____init__方法,因此要创建自己的元类需要继承type类并通常覆盖__new____init__方法
    • new: 创建类对象
    • init: 初始化类对象
    • prepare: 返回用于类属性初始化的名称空间字典
class MyMeta(type):@classmethoddef __prepare__(cls, name, bases, **kwargs):"""在类创建前准备命名空间返回一个用于存储类属性的字典-like对象"""print(f"1. __prepare__: 准备 {name} 的命名空间")# 这里使用普通字典,但可以返回自定义的字典子类# 例如 OrderedDict 保持属性顺序return {'__annotations__': {}}def __new__(mcls, name, bases, namespace, **kwargs):"""实际创建类对象可以修改类的属性或基类"""print(f"2. __new__: 创建 {name} 类对象")# 自动添加类创建时间戳namespace['created_at'] = '2023-11-01'# 强制添加前缀到非魔术方法属性名new_namespace = {f'custom_{k}' if not k.startswith('__') else k: vfor k, v in namespace.items()}return super().__new__(mcls, name, bases, new_namespace)def __init__(cls, name, bases, namespace, **kwargs):"""初始化已创建的类可以添加类初始化逻辑"""print(f"3. __init__: 初始化 {name} 类")super().__init__(name, bases, namespace)# 添加类方法cls.class_info = lambda self: f"Class {name} created with MyMeta"

4、使用元类

使用metaclass参数指定要继承的元类:

class MyClass(metaclass=MyMeta):"""使用自定义元类的示例类"""value = 42def __init__(self, x):self.x = xdef method(self):return self.x# 测试输出
print("\n类创建完成,开始实例化测试:")
obj = MyClass(10)
print(f"访问修改后的属性: value -> {hasattr(MyClass, 'value')}")  # False
print(f"访问添加的前缀属性: custom_value -> {MyClass.custom_value}")  # 42
print(f"访问自动添加的属性: created_at -> {MyClass.created_at}")  # 2023-11-01
print(f"调用添加的方法: {obj.class_info()}")  # Class MyClass created with MyMeta

5、元类的主要用途

  • 控制类的创建行为:可以在类创建时修改类属性
  • 自动注册子类:框架中自动发现和注册插件
  • 强制接口实现:确保子类实现了特定方法
    ORM实现:如Django的Model、SQLAlchemy的DeclarativeBase

6、元类的实际应用

  • 注册所有子类
class PluginMeta(type):def __init__(cls, name, bases, namespace):super().__init__(name, bases, namespace)if not hasattr(cls, 'plugins'):cls.plugins = []else:cls.plugins.append(cls)class Plugin(metaclass=PluginMeta):passclass PluginA(Plugin):passclass PluginB(Plugin):passprint(Plugin.plugins)  # 输出: [<class '__main__.PluginA'>, <class '__main__.PluginB'>]
  • 强制方法实现
class AbstractMeta(type):def __new__(cls, name, bases, namespace):if 'required_method' not in namespace:raise TypeError(f"{name} must implement 'required_method'")return super().__new__(cls, name, bases, namespace)class AbstractClass(metaclass=AbstractMeta):passclass ValidClass(AbstractClass):def required_method(self):passclass InvalidClass(AbstractClass):  # 会引发TypeErrorpass
  • 自动添加属性
class AutoAttributeMeta(type):def __new__(cls, name, bases, namespace):namespace['auto_attr'] = f"auto_{name.lower()}"return super().__new__(cls, name, bases, namespace)class MyClass(metaclass=AutoAttributeMeta):passprint(MyClass.auto_attr)  # 输出: auto_myclass

动态类

1、什么是动态类

  • 动态类是指在程序运行时(而非编写代码时)动态创建的类。
  • 动态类具备根据需要修改类的结构和行为的灵活性
  • 动态类的元编程能力利用了python的反射和元类机制

2、使用type()函数动态创建类

  • type的三个参数:
    1. 类名
    2. 基类元组
    3. 包含属性的字典
  • 动态创建类的基本方法: type(类名, 基类元组, 属性字典)

DynamicClass = type('DynamicClass', (), {'x': 10,'say_hello': lambda self: f"Hello! x is {self.x}"
})
# 这个类的创建等价于:
# class DynamicClass:  静态定义
# 	 'x': 10,
#    'say_hello': lambda self: f"Hello! x is {self.x}"obj = DynamicClass()
print(obj.say_hello())  # 输出: Hello! x is 10
  • 在类定义后,为类动态添加属性和方法
class Person:pass# 动态添加方法
def greet(self):return f"Hello, I'm {self.name}"Person.greet1 = greetp = Person()
p.name = "Alice"
print(p.greet1())  # 输出: Hello, I'm Alice
p.__dict__ # 输出:{'name': 'Alice'},表示这个实例只有name属性
  • 使用 types.new_class() 创建(Python 3.3+)
from types import new_classdef __init__(self, value):self.value = valueDynamicClass = new_class('DynamicClass',(object,),exec_body=lambda ns: ns.update({'__init__': __init__,'get_value': lambda self: self.value})
)obj = DynamicClass(42)
print(obj.get_value())  # 输出: 42

3、动态类的优势

  • (1) 运行时决定类结构
def create_class(has_method):attrs = {'x': 10}if has_method:attrs['method'] = lambda self: "Dynamic!"return type('DynamicClass', (), attrs)ClassWithMethod = create_class(True)
ClassWithoutMethod = create_class(False)
  • (2) 动态继承
BaseClass = SomeClass if condition else AnotherClass
DynamicClass = type('DynamicClass', (BaseClass,), {})
  • (3) 与元编程结合
class Meta(type):def __new__(cls, name, bases, attrs):attrs['custom'] = "Added by metaclass"return super().__new__(cls, name, bases, attrs)# 动态使用元类
DynamicClass = type('DynamicClass', (), {'__metaclass__': Meta})

4、动态类的实际应用场景

  • 插件系统:运行时加载和注册插件类

  • ORM 映射:根据数据库表结构动态生成模型类

  • 协议适配:根据不同的协议规范生成处理类

  • 测试替身:动态创建模拟类(Mock)用于测试

  • DSL 实现:领域特定语言的运行时类生成

5、注意事项

  • 可读性:过度使用动态类会使代码难以维护

  • 性能:动态创建类比静态定义稍慢

  • 调试:动态生成的类可能更难调试

  • 文档:需要额外文档说明动态行为


http://www.hkcw.cn/article/vxbqfRWkNI.shtml

相关文章

SpringBoot整合RocketMQ--实例

原文网址&#xff1a;SpringBoot整合RocketMQ--实例-CSDN博客 简介 本文介绍SpringBoot整合RocketMQ的方法。 spring-boot-starter-parent版本&#xff1a;2.4.13RocketMQ版本&#xff1a;4.9.4。&#xff08;写这篇文章时&#xff0c;5.X版本的Java客户端还没完善&#xff…

9.5 Q1 | 北京协和医学院GBD发文 | 1990-2021 年全球、区域和国家心力衰竭负担及其根本原因

1.第一段-文章基本信息 文章题目&#xff1a;Global, regional, and national burden of heart failure and its underlying causes, 1990-2021: results from the global burden of disease study 2021 中文标题&#xff1a;1990-2021 年全球、区域和国家心力衰竭负担及其根本…

汇聚全球智慧,共话艺术设计与现代化教育——ADME 2025

会议简介 第二届艺术设计与现代化教育国际会议&#xff08;ADEME 2025&#xff09;在风光旖旎的春城昆明隆重召开。这是一场集全球艺术设计精英与教育创新者于一体的学术盛宴。会议围绕“创意启迪教育革新”主题&#xff0c;旨在搭建一个多元文化交流与知识共享的平台&#xff…

从 SWT Browser 迁移到 JxBrowser

多年来&#xff0c;SWT 一直内置一个 Browser 组件。这是一个依赖于操作系统自带的 Web engine 的简单组件。该组件可以很好地显示网页并处理简单的任务&#xff0c;但对于需要跨平台行为一致、更好地控制 Engine、隔离用户数据等更高级需求来说&#xff0c;它显然不够用。 因…

编译原理笔记 2025/4/22

基本概念 汇编语言与高级程序设计语言的关系/汇编干嘛的&#xff1a;高级语言与硬件无关&#xff0c;汇编语言的定义与CPU的指令系统直接相关。只要将高级语言编写的程序等价地转换成特定硬件平台所支持的方式来实现&#xff08;汇编程序或机器指令序列&#xff09;&#xff0…

(ICML-2025) RIFLEx:视频扩散Transformer中长度外推的“免费午餐”

RIFLEx&#xff1a;视频扩散Transformer中长度外推的“免费午餐” paper title&#xff1a;RIFLEx: A Free Lunch for Length Extrapolation in Video Diffusion Transformers paper是THU发表在ICML 2025的工作 Code:链接 Abstract 近期视频生成的进展使模型能够合成高质量的分…

树莓派超全系列教程文档--(52)如何启用VNC功能

如何启用VNC功能 使用 VNC 共享屏幕启用 VNC 服务器以图形方式启用 VNC 服务器在命令行上启用 VNC 服务器 连接到 VNC 服务器 文章来源&#xff1a; http://raspberry.dns8844.cn/documentation 原文网址 使用 VNC 共享屏幕 有时&#xff0c;使用设备进行物理操作并不方便。…

TDengine 运维——巡检工具(安装工具)

背景 TDengine 的安装包自带安装脚本&#xff0c;但无法基于集群进行自动化安装部署&#xff0c;本文档旨在说明如何使用安装工具进行 TDengine 的集群式安装部署。 安装工具支持功能 安装方式详细说明单节点安装部署单节点环境安装部署 TDengine集群安装部署集群环境安装部…

Qt Creator调用Python代码

Qt Creator下调用Python代码 在Qt编写的上位机,现在可能经常用到Python相关的代码。本篇记录Qt Creator中调用Python的一种方法。 Python使用的版本为 3.9.10,(安装参考:Python3.9的安装和配置) Qt 使用的版本为5.14.2,(Qt的安装可以参考网上的安装案例:Qt 5.14安装…

政策+技术双轮驱动:MiC建筑如何成为“好房子”建设的破局之道

在建筑行业不断追求创新与可持续发展的今天&#xff0c;模块化集成建筑&#xff08;Modular Integrated Construction&#xff0c;简称MiC&#xff09;正逐渐崭露头角&#xff0c;成为推动行业转型升级的重要力量。近日&#xff0c;全国政协常委、人口资源环境委员会副主任&…

Python Day37

Task&#xff1a; 1.过拟合的判断&#xff1a;测试集和训练集同步打印指标 2.模型的保存和加载 a.仅保存权重 b.保存权重和模型 c.保存全部信息checkpoint&#xff0c;还包含训练状态 3.早停策略 1. 过拟合的判断&#xff1a;测试集和训练集同步打印指标 过拟合是指模型在训…

2025年全国青少年信息素养大赛 scratch图形化编程挑战赛 小低组初赛 内部集训模拟题解析

2025年信息素养大赛初赛scratch模拟题解析 博主推荐 所有考级比赛学习相关资料合集【推荐收藏】 scratch资料 Scratch3.0系列视频课程资料零基础学习scratch3.0【入门教学 免费】零基础学习scratch3.0【视频教程 114节 免费】 历届蓝桥杯scratch国赛真题解析历届蓝桥杯scr…

Linux环境基础开发工具->gcc/g++

引入&#xff1a;gcc/g是什么&#xff1f; 在上篇博客我们知道&#xff0c;vim是一个编辑器&#xff0c;vim负责的是代码的编辑&#xff1b;而gcc/g是一个编译器&#xff0c;负责的就是代码的编译&#xff01;gcc负责C语言代码的编译&#xff0c;而g负责c代码的编译&#xff0…

云原生与DevOps融合实践:加速企业数字化转型的加速器

&#x1f4dd;个人主页&#x1f339;&#xff1a;一ge科研小菜鸡-CSDN博客 &#x1f339;&#x1f339;期待您的关注 &#x1f339;&#x1f339; 一、引言&#xff1a;为什么“云原生DevOps”是当下最强组合&#xff1f; 在传统软件交付模式逐步被淘汰的当下&#xff0c;越来…

孙颖莎王曼昱出战WTT美国站女双 拉斯维加斯再携手

2025年WTT美国大满贯将于7月3日至13日在拉斯维加斯奥尔良体育馆及美高梅大酒店会议中心举行。孙颖莎和王曼昱将搭档出战女双正赛。在不久前结束的多哈世乒赛女单决赛中,孙颖莎以4比3的大比分险胜王曼昱,成功卫冕。责任编辑:zx0176

基于51单片机和8X8点阵屏、独立按键的射击消除类小游戏

目录 系列文章目录前言一、效果展示二、原理分析三、各模块代码1、8X8点阵屏2、独立按键3、定时器04、定时器1 四、主函数总结 系列文章目录 前言 使用的是普中A2开发板。 【单片机】STC89C52RC 【频率】12T11.0592MHz 【外设】8X8点阵屏、独立按键 效果查看/操作演示&#x…

ubuntu22.04安装docker

1. 准备工作 更新系统软件包索引 sudo apt update2. 卸载旧版本 Docker&#xff08;可选&#xff09; 清理旧版 Docker 及相关依赖 sudo apt-get remove docker docker-engine docker.io containerd runc3. 设置 Docker 仓库 安装依赖工具 (apt-transport-https, ca-certi…

burpsuit抓包完整示例

1.确保浏览器&#xff08;这里使用的是火狐浏览器&#xff09;和burpsuit配置完整&#xff08;有需要留言&#xff09;&#xff0c;配置完整包括jdk安装&#xff0c;配置环境变量&#xff0c;下载burp,下载并导入证书&#xff0c;ip端口一致&#xff0c;代理能正常打开。 2.注意…

其他 | 边缘端应用的轻量级优化调研

1.调研目标 由于边缘计算场景的性能受限&#xff0c;无法提供与常规服务器相同或略低的环境&#xff0c;因此对我们的上层业务应用有着较高的资源要求。 目前我们的应用程序基于 Oracle JDK&#xff08;开发者端&#xff09;与 OpenJDK&#xff08;生产环境&#xff09;进行开…

Shell 脚本常用命令笔记

一、系统配置命令 1. 主机名设置 文件方式 修改文件&#xff1a;vim /etc/hostname&#xff0c;写入新主机名&#xff08;如czg.easylee.org&#xff09;。生效方式&#xff1a;需重新打开 Shell 或重启系统。 命令方式 即时生效命令&#xff1a;hostnamectl set-hostname 新…