流畅的Python(二十一)-类元编程

一、核心要义

1.类元编程时指在运行时创建或定制类的技艺

2.类是一等对象,因此任何时候都可以使用函数新建类,而无需使用class关键字

3.类装饰器也是函数,不过能够审查、修改,甚至把被装饰的类替换为其它类。

4.元类(type类的子类)类编程最高级的工具:使用元类可以创建具有某种特质的全新类种,例如我们见过的抽象基类。

5.所有类都是type类的实例,但只有元类既是type类的实例,也是其子类。

温馨提示:除非开发框架,否则不要编写元类。

二、代码示例

1、类工厂函数

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time    : 2024/3/17 18:24
# @Author  : Maple
# @File    : 01-类工厂函数.py
# @Software: PyCharmdef record_factory(cls_name,field_names):try:field_names = field_names.replace(',',' ').split(' ')except AttributeError:passfield_names = tuple(field_names)def __init__(self,*args,**kwargs):# 属性键值对:处理元组参数attrs = dict(zip(self.__slots__,args))# 属性键值对:处理key-value参数attrs.update(**kwargs)for name,value in attrs.items():# 为实例对象设置属性setattr(self,name,value)def __iter__(self):for name in self.__slots__:yield getattr(self,name)def __repr__(self):# 注意zip(self.__slots__,self)中后面一个self,其实是一个元组(各个属性对应的值)遍历,因为实例实现了__iter__方法,values = ', '.join('{}={!r}'.format(*i) for i in zip(self.__slots__,self))return '{}({})'.format(self.__class__.name,values)# 组件类属性字典cls_attrs = dict(__slots__ = field_names,__init__ = __init__,__iter__ = __iter__,__repr__ = __repr__)# 返回一个类return type(cls_name,(object,),cls_attrs)if __name__ == '__main__':# 1.创建一个Dog的类(也是一个对象-万物皆对象)Dog1 = record_factory('Dog','name,age,gender')d1 = Dog1('旺财',18,'母') # <member 'name' of 'Dog' objects>(name='旺财', age=18, gender='母')print(d1) # <member 'name' of 'Dog' objects>(name='旺财', age=18, gender='母')# 2.创建另外一个Dog类(也是一个对象-万物皆对象)Dog2 = record_factory('Dog','name,age,gender')d2 = Dog2('阿来', 20, '公')print(d2) # <member 'name' of 'Dog' objects>(name='阿来', age=20, gender='公')# 3.Dog2和Dog1不是同一个对象print(id(Dog2) == id(Dog1)) #Falseprint(Dog1 == Dog2) # False

2、类工厂函数

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time    : 2024/3/17 20:54
# @Author  : Maple
# @File    : 02-定制描述符的类装饰器.py
# @Software: PyCharmimport abcdef entity(cls):#keyde的值类似:weight#attr的值类似:<__main__.Quantity object at 0x00000262A8BBA3A0>,是一个描述符对象for key,attr in cls.__dict__.items():if isinstance(attr,Validated):# type_name类似:Quantitytype_name = type(attr).__name__attr.storage_name = '_{}#{}'.format(type_name,key)return clsclass AutoStorage:__counter = 0def __init__(self):cls = self.__class__prefix = cls.__name__index = cls.__counterself.storage_name = '_{}#{}'.format(prefix,index)cls.__counter += 1def __get__(self, instance, owner):if instance is None:return selfelse:return getattr(instance,self.storage_name)def __set__(self, instance, value):setattr(instance,self.storage_name,value)class Validated(abc.ABC,AutoStorage):def __set__(self, instance, value):value = self.validate(instance,value)super().__set__(instance,value)@abc.abstractmethoddef validate(self,instance,value):"""retuen validated value or raise ValueError"""class Quantity(Validated):"""a number greater than zero"""def validate(self,instance,value):if value > 0:return valueelse:raise ValueError('value must be >0')class NonBlank(Validated):"""a string with at least one non-space character"""def validate(self,instance,value):value = value.strip()if len(value) == 0:raise ValueError('value can not be empty or blank')else:return value# 用entity装饰LineItem类
# 默认的属性名变成类似_Quantity#weight和_Quantity#price,_NonBlank#price
@entity
class LineItem:description = NonBlank()weight = Quantity()price = Quantity()def __init__(self,description,weight,price):self.description = descriptionself.weight = weightself.price = pricedef subtotal(self):return self.weight * self.pricedef __repr__(self):return 'LineItem =({},{},{})'.format(self.description,self.weight,self.price)if __name__ == '__main__':com = LineItem('Computer', 10, 1000)# 查看实例属性的名print(com.__dict__) # {'_NonBlank#description': 'Computer', '_Quantity#weight': 10, '_Quantity#price': 1000}

3、导入时和运行时比较

(1) evalsupport.py

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time    : 2024/3/18 21:03
# @Author  : Maple
# @File    : evalsupport.py
# @Software: PyCharmprint('<[100]> evalsupport module start')def deco_alpha(cls):print('<[200]> deco_alpha')def innner_1(self):print('<[300]> deco_alpha:inner_1')cls.method_y  = innner_1return  cls# 注意MetaAleph继承type,因此它是一个元类
class MetaAleph(type):print('<[400]> MetaAleph body')def __init__(cls,name,bases,dict):print('<500> MetaAleph.__init__')def inner_2(self):print('<600> MetaAleph.__init__:inner_2')cls.method_z = inner_2print('<[700]> evalsupport module end')

(2) evaltime.py

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time    : 2024/3/18 20:51
# @Author  : Maple
# @File    : evaltime.py
# @Software: PyCharmfrom evalsupport import deco_alphaprint('<[1]> evaltime module start')class ClassOne(object):print('<[2]> ClassOne body')def __init__(self):print('<[3]> ClassOne.__init__')def __del__(self):print('<[4]> ClassOne.__del__')def method_x(self):print('<[5]> ClassOne.method_x')class ClassTwo(object):print('<[6]> ClassTwo body')@deco_alpha
class ClassThree(object):print('<[7]> ClassThree body')def method_y(self):print('<[8]> ClassThree.method_y')class ClassFour(object):print('<[9]> ClassFour body')def method_y(self):print('<[10]> ClassFour.method_y')if __name__ == '__main__':print('<[11]> ClassOne tests', 30 * '.')one = ClassOne()one.method_x()print('<[12]> ClassThree tests', 30 * '.')three = ClassThree()three.method_y()print('<[13]> ClassFour tests', 30 * '.')four = ClassFour()four.method_y()print('<[14]> evaltime module end')

(3) 导入时

① 打开Python Console窗口

② 执行import evaltime

Tips:此时evaltime.py文件的name = __main__部分并没有执行

(4) 运行时

注意:在Teminal窗口执行

重点关注部分:

No1:因为@deco_alpha装饰器修改了ClassThree的method_y方法为inner_1,因此当three调用method_1的时候,inner_1方法会调用,并输出<[300]> deco_alpha:inner_1

No2: 虽然ClassFour继承了ClassThree,但是@deco_alpha装饰器并不会作用在ClassFour上,因此four.method_y的时候,并没有输出<[300]> deco_alpha:inner_1

4、元类的运行顺序

(1)evaltime_meta.py

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time    : 2024/3/18 21:03
# @Author  : Maple
# @File    : evalsupport.py
# @Software: PyCharmprint('<[100]> evalsupport module start')def deco_alpha(cls):print('<[200]> deco_alpha')def innner_1(self):print('<[300]> deco_alpha:inner_1')cls.method_y  = innner_1return  cls# 注意MetaAleph继承type,因此它是一个元类
class MetaAleph(type):print('<[400]> MetaAleph body')def __init__(cls,name,bases,dict):print('<500> MetaAleph.__init__')def inner_2(self):print('<600> MetaAleph.__init__:inner_2')cls.method_z = inner_2print('<[700]> evalsupport module end')

(2)导入时

重点说明

    1. 创建ClassFive时,调用了MetaAleph.__init__方法

    2. 创建ClassFive的子类ClassSix时,同样会调用MetaAleph.__init__方法

(3)运行时

要点:元类会改变被修饰的类的一些特性

和装饰器@deco_alpha的区别是,MetaAleph的作用在ClassFive的子类ClassSix种仍然起作用。

5、定制描述符的元类

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time    : 2024/3/19 20:51
# @Author  : Maple
# @File    : 04-定制描述符的元类.py
# @Software: PyCharmimport abcclass AutoStorage:__counter = 0def __init__(self):cls = self.__class__prefix = cls.__name__index = cls.__counterself.storage_name = '_{}#{}'.format(prefix,index)cls.__counter += 1def __get__(self, instance, owner):if instance is None:return selfelse:return getattr(instance,self.storage_name)def __set__(self, instance, value):setattr(instance,self.storage_name,value)class Validated(abc.ABC,AutoStorage):def __set__(self, instance, value):value = self.validate(instance,value)super().__set__(instance,value)@abc.abstractmethoddef validate(self,instance,value):"""retuen validated value or raise ValueError"""class Quantity(Validated):"""a number greater than zero"""def validate(self,instance,value):if value > 0:return valueelse:raise ValueError('value must be >0')class NonBlank(Validated):"""a string with at least one non-space character"""def validate(self,instance,value):value = value.strip()if len(value) == 0:raise ValueError('value can not be empty or blank')else:return valueclass EntityMeta(type):def __init__(cls, name, bases, attr_dict):super().__init__(name, bases, attr_dict)for key, attr in attr_dict.items():if isinstance(attr, Validated):type_name = type(attr).__name__attr.storage_name = '_{}.{}'.format(type_name, key)class Entity(metaclass=EntityMeta):"""带有验证字段的业务实体"""# LineItem是 Entity 的子类,而子类可以继承父类的元类特性
# 因此LineItem类也会被EntityMeta管控
class LineItem(Entity):description = NonBlank()weight = Quantity()price = Quantity()def __init__(self,description,weight,price):self.description = descriptionself.weight = weightself.price = pricedef subtotal(self):return self.weight * self.pricedef __repr__(self):return 'LineItem =({},{},{})'.format(self.description,self.weight,self.price)if __name__ == '__main__':com = LineItem('Computer', 10, 1000)# 查看实例属性的名print(com.__dict__)  # {'_NonBlank#description': 'Computer', '_Quantity#weight': 10, '_Quantity#price': 1000}

6、元类的prepare方法

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time    : 2024/3/19 21:10
# @Author  : Maple
# @File    : 05-元类的prepare方法.py
# @Software: PyCharmimport abc
import collectionsclass AutoStorage:__counter = 0def __init__(self):cls = self.__class__prefix = cls.__name__index = cls.__counterself.storage_name = '_{}#{}'.format(prefix, index)cls.__counter += 1def __get__(self, instance, owner):if instance is None:return selfelse:return getattr(instance, self.storage_name)def __set__(self, instance, value):setattr(instance, self.storage_name, value)class Validated(abc.ABC, AutoStorage):def __set__(self, instance, value):value = self.validate(instance, value)super().__set__(instance, value)@abc.abstractmethoddef validate(self, instance, value):"""retuen validated value or raise ValueError"""class Quantity(Validated):"""a number greater than zero"""def validate(self, instance, value):if value > 0:return valueelse:raise ValueError('value must be >0')class NonBlank(Validated):"""a string with at least one non-space character"""def validate(self, instance, value):value = value.strip()if len(value) == 0:raise ValueError('value can not be empty or blank')else:return valueclass EntityMeta(type):@classmethoddef __prepare__(metacls, name, bases):# 1.该方法先于new和init方法执行# 2.该方法必须返回一个映射# 3.该方法的返回值会传递给init方法的最后一个参数(本例是attr_dict)return collections.OrderedDict()def __init__(cls, name, bases, attr_dict):# 注意:此时attr_dict是OrderedDict对象,其特点是先进先出super().__init__(name, bases, attr_dict)# 为类添加一个类属性cls._filed_names = []for key, attr in attr_dict.items():if isinstance(attr, Validated):type_name = type(attr).__name__attr.storage_name = '_{}.{}'.format(type_name, key)cls._filed_names.append(key)class Entity(metaclass=EntityMeta):"""带有验证字段的业务实体"""@classmethoddef field_names(cls):for key in cls._filed_names:yield key# LineItem是 Entity 的子类,而子类可以继承父类的元类特性
# 因此LineItem类也会被EntityMeta管控
class LineItem(Entity):description = NonBlank()weight = Quantity()price = Quantity()def __init__(self, description, weight, price):self.description = descriptionself.weight = weightself.price = pricedef subtotal(self):return self.weight * self.pricedef __repr__(self):return 'LineItem =({},{},{})'.format(self.description, self.weight, self.price)if __name__ == '__main__':com = LineItem('Computer', 10, 1000)# 查看实例属性的名print(com.__dict__)  # {'_NonBlank#description': 'Computer', '_Quantity#weight': 10, '_Quantity#price': 1000}# 按照添加字段的顺序产出字段的名字for name in LineItem.field_names():"""descriptionweightprice"""print(name)

三、结语

历史将近4个月,《流畅的Python》完结撒花, 下一站《Python数据科学》,相聚有时,后会有期。

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

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

相关文章

FDU 2019 | 3. 有向树形态

文章目录 1. 题目描述2. 我的尝试1. 卡特兰数2. 动态规划 1. 题目描述 求 N 个结点能够组成的二叉树的个数。 输入格式 一个整数 N 。 输出格式 输出能组成的二叉树的个数。 数据范围 1 ≤ N ≤ 20 1≤N≤20 1≤N≤20 输入样例 3输出样例 52. 我的尝试 1. 卡特兰数 直接…

分布式锁的详细解释

什么是分布式锁 分布式锁是一种用于协调分布式系统中多个进程或线程之间访问共享资源的机制。在分布式系统中&#xff0c;多个进程或线程可能同时竞争访问某个共享资源&#xff0c;为了避免并发访问导致的数据不一致或冲突&#xff0c;需要使用分布式锁来保证资源的独占性。 分…

5.1.1、【AI技术新纪元:Spring AI解码】Openai chat

OpenAI 聊天 Spring AI 支持由 OpenAI 开发的人工智能语言模型 ChatGPT。ChatGPT 在激发人们对 AI 驱动文本生成的兴趣方面发挥了重要作用,这归功于它创建的行业领先的文本生成模型和嵌入技术。 先决条件 您需要与 OpenAI 创建一个 API 来访问 ChatGPT 模型。在 OpenAI 注册页…

第十三届蓝桥杯省赛CC++ 研究生组

蓝桥杯2022年第十三届省赛真题-裁纸刀 蓝桥杯2022年第十三届省赛真题-灭鼠先锋 蓝桥杯2022年第十三届省赛真题-质因数个数 求个数&#xff0c;则只需要计数即可。求啥算啥&#xff0c;尽量不要搞多余操作 蓝桥杯2022年第十三届省赛真题-选数异或 蓝桥杯2022年第十三届省赛真题…

深入理解并优化Android中的文件描述符(FD)

一、文件描述符&#xff08;FD&#xff09;概述 文件描述符&#xff08;File Descriptor&#xff0c;简称FD&#xff09;是Unix和类Unix操作系统&#xff08;包括Android&#xff09;中的一个关键概念。它是一个非负整数&#xff0c;用于标识操作系统分配的文件或其他输入/输出…

内网渗透学习-环境搭建

1、环境搭建测试 虚拟机网络环境配置&#xff0c;模拟外网和内网 主机操作系统网络内网ip外网ip物理主机window10vmnet8192.168.70.1攻击机kali Linuxvmnet8192.168.70.134域控主机win server 2008 r2vmnet0192.168.52.138域成员主机win server 2k3vmnet0192.168.52.141服务器…

HarmonyOS NEXT应用开发之swiper指示器导航点位于swiper下方

介绍 本示例介绍通过分割swiper区域&#xff0c;实现指示器导航点位于swiper下方的效果。 效果预览图 使用说明 加载完成后swiper指示器导航点&#xff0c;位于显示内容下方。 实现思路 将swiper区域分割为两块区域&#xff0c;上方为内容区域&#xff0c;下方为空白区域。…

el-input添加keyup事件无响应

<el-input type"password" v-model"formData.password" keyup.enter"onSubmit"></el-input>使用 .native修饰符 有时&#xff0c;Vue 组件内部可能会处理某些事件&#xff0c;导致你不能直接在组件上监听这些事件。 在这种情况下&am…

【代码随想录Day28】

Day 28 回溯 Part04 今日任务 93.复原IP地址78.子集90.子集II 代码实现 今天的题出乎意料的简单 复原IP地址&#xff0c;和昨天的分割回文串比较像 一个是分割&#xff0c;一个是判断ip有效 List<String> result new ArrayList<>();List<String> path …

【数据结构】链表力扣刷题详解

前言 题目链接 移除链表元素 链表的中间结点 反转链表 分割链表 环形链表的约瑟夫问题 ​ 欢迎关注个人主页&#xff1a;逸狼 创造不易&#xff0c;可以点点赞吗~ 如有错误&#xff0c;欢迎指出~ 移除链表元素 题述 给你一个链表的头节点 head 和一个整数 val &#xff0c;请…

【Spring Cloud Gateway】路由配置uri三种方式及区别

websocket配置方式 ws:// 或 wss:// 开头的 URI&#xff0c;表示配置的是支持 Websocket 协议的目标地址。 这种方式适用于需要与客户端建立长连接、实现双向通信的场景&#xff0c;比如实时消息推送、即时聊天等。 使用 Websocket 配置方式可以让 Spring Cloud Gateway 能够…

从基础入门到学穿C++

前言知识 C简介 C是一门什么样的语言&#xff0c;它与C语言有着什么样的关系&#xff1f; C语言是结构化和模块化的语言&#xff0c;适合处理较小规模的程序。对于复杂的问题&#xff0c;规模较大的程序&#xff0c;需要高度的抽象和建模时&#xff0c;C语言则不合适。为了解…

代理IP品质对Tik Tok代理的重要性

随着Tik Tok的迅速崛起&#xff0c;越来越多的人开始关注如何透过Tik Tok进行行销和推广。其中&#xff0c;使用Tik Tok代理程式是常见的方法。 然而&#xff0c;在选择和使用代理时&#xff0c;IP品质是一个不可忽视的因素。本文将探讨IP品质对Tik Tok代理的重要性&#xff0…

鸿蒙Harmony应用开发—ArkTS声明式开发(容器组件:GridRow)

栅格布局可以为布局提供规律性的结构&#xff0c;解决多尺寸多设备的动态布局问题&#xff0c;保证不同设备上各个模块的布局一致性。 栅格容器组件&#xff0c;仅可以和栅格子组件(GridCol)在栅格布局场景中使用。 说明&#xff1a; 该组件从API Version 9开始支持。后续版本…

初中分班怎么分按什么标准

初中是学生生涯中一个重要的阶段&#xff0c;它承接着小学的基础教育&#xff0c;同时为高中的深入学习打下基础。在这个关键时期&#xff0c;分班成为学校、家长和学生共同关注的焦点。那么&#xff0c;初中分班是按照什么标准来进行的呢&#xff1f; 学业成绩是初中分班的首要…

HarmonyOS 网络请求工具库封装,直接无脑用!!!

前言 HarmonyOS 原生网络请求的用法比较麻烦&#xff0c;还是有必要封装下&#xff0c;先看它的原生写法&#xff1a; // 引入包名 import http from ohos.net.http;// 每一个httpRequest对应一个HTTP请求任务&#xff0c;不可复用 let httpRequest http.createHttp(); // 用…

考研复习C语言进阶(4)

1. 为什么存在动态内存分配 我们已经掌握的内存开辟方式有&#xff1a; int val 20;//在栈空间上开辟四个字节 char arr[10] {0};//在栈空间上开辟10个字节的连续空间 但是上述的开辟空间的方式有两个特点&#xff1a; 1. 空间开辟大小是固定的。 2. 数组在申明的时候&#…

同步方法和同步块,哪个是更好的选择?什么是线程同步和线程互斥,有哪几种实现方式?

该文章专注于面试,面试只要回答关键点即可,不需要对框架有非常深入的回答,如果你想应付面试,是足够了,抓住关键点 同步方法和同步块,哪个是更好的选择 在 Java 中,同步方法和同步块都是用于保护共享资源、避免线程之间互相干扰的机制。它们都可以用来实现线程安全的操…

Java基础---映射框架

1.数据结构 数组 特点:存储区间连续的&#xff0c;内存占用严重优点:随机读取和修改效率高。 查询来说时间复杂度O(1)缺点:插入和删除速度慢。时间复杂度为O(N) 链表 特点:存储区间是离散的。内存利用率高优点:插入和删除速度快缺点&#xff1a;不能随机查找。查询效率低 …

Docker知识--01

虚拟化 # 什么是虚拟化 在计算机中&#xff0c;虚拟化&#xff08;英语&#xff1a;Virtualization&#xff09;是一种资源管理技术&#xff0c;是将计算机的各种实体资源&#xff0c;如服务器、网络、内存及存储等&#xff0c;予以抽象、转换后呈现出来&#xff0c;打…