python元类为类的全部方法添加装饰器

2024-01-23 09:52

本文主要是介绍python元类为类的全部方法添加装饰器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1 python元类为类的全部方法添加装饰器

装饰器和元类都在class语句的末尾运行,同时使用装饰器和元类实现对一个类的所有方法应用一个函数装饰器。

1.1 调用次数和计时函数

示例

>>> import time,sys
>>> def tracer(func):calls=0def onCall(*args,**kargs):nonlocal callscalls+=1print('调用 %s %s 次' % (func.__name__,calls))return func(*args,**kargs)return onCall>>> def timer(label='',trace=True):def onDecorator(func):def onCall(*args,**kargs):beg=time.perf_counter()res=func(*args,**kargs)usetime=time.perf_counter()-begonCall.alltime+=usetimeif trace:timestr='{0[0]}:usetime={0[1]:.6f},alltime={0[2]:.6f}'timeval=func.__name__,usetime,onCall.alltimeprint(label,timestr.format(timeval))return resonCall.alltime=0return onCallreturn onDecorator

1.2 用装饰器手动统计调用次数

描述

在需要统计调用次数和计时的每个方法前手动编写@装饰语法。

示例

>>> class Staff:@tracerdef __init__(self,name,pay):self.name=nameself.pay=pay@tracerdef giveRaise(self,rate):self.pay*=(1.0+rate)@tracerdef lastName(self):return self.name[0]>>> s1=Staff('张三',50000)
调用 __init__ 1 次
>>> s2=Staff('李四',51000)
调用 __init__ 2 次
>>> print(s1.name,s2.name)
张三 李四
>>> s1.giveRaise(0.1)
调用 giveRaise 1 次
>>> s1.pay
55000.00000000001
>>> print(s1.lastName(),s2.lastName())
调用 lastName 1 次
调用 lastName 2 次
张 李

1.3 用元类和装饰器统计调用次数

描述

手动使用装饰器使用于需装饰方法数量较少的情况。如果需对一个类的全部方法进行装饰,则通过元类将装饰器添加到每个方法中,达到自动装饰的效果。

示例

>>> from types import FunctionType
>>> class MetaTrace(type):def __new__(meta,classname,supers,classdict):for attr,attrval in classdict.items():# 类属性为方法则自动装饰if type(attrval) is FunctionType:# 调用装饰器 trace 重新绑定装饰方法classdict[attr]=tracer(attrval)return type.__new__(meta,classname,supers,classdict)>>> class Staff(metaclass=MetaTrace):def __init__(self,name,pay):self.name=nameself.pay=paydef giveRaise(self,rate):self.pay*=(1.0+rate)def lastName(self):return self.name[0]>>> s1=Staff('张三',50000)
调用 __init__ 1 次
>>> s2=Staff('李四',51000)
调用 __init__ 2 次
>>> print(s1.name,s2.name)
张三 李四
>>> s1.giveRaise(0.1)
调用 giveRaise 1 次
>>> s1.pay
55000.00000000001
>>> print(s1.lastName(),s2.lastName())
调用 lastName 1 次
调用 lastName 2 次
张 李

1.4 把任何装饰器应用于方法

描述

通过元类把传入的装饰器应用到一个类的所有方法,只需在原来外层定义一个函数接收传入队装饰器即可。

示例

>>> from types import FunctionType
>>> def decoratorAll(decorator):class MetaTrace(type):def __new__(meta,classname,supers,classdict):for attr,attrval in classdict.items():if type(attrval) is FunctionType:classdict[attr]=decorator(attrval)return type.__new__(meta,classname,supers,classdict)return MetaTrace
>>> class Staff(metaclass=decoratorAll(tracer)):def __init__(self,name,pay):self.name=nameself.pay=paydef giveRaise(self,rate):self.pay*=(1.0+rate)def lastName(self):return self.name[0]>>> s1=Staff('张三',50000)
调用 __init__ 1 次
>>> s2=Staff('李四',51000)
调用 __init__ 2 次
>>> print(s1.name,s2.name)
张三 李四
>>> s1.giveRaise(0.1)
调用 giveRaise 1 次
>>> s1.pay
55000.00000000001
>>> print(s1.lastName(),s2.lastName())
调用 lastName 1 次
调用 lastName 2 次
张 李
# 使用timer装饰器
>>> class Staff(metaclass=decoratorAll(timer(label='**'))):def __init__(self,name,pay):self.name=nameself.pay=paydef giveRaise(self,rate):self.pay*=(1.0+rate)def lastName(self):return self.name[0]>>> s1=Staff('张三',50000)
** __init__:usetime=0.000003,alltime=0.000003
>>> s2=Staff('李四',51000)
** __init__:usetime=0.000002,alltime=0.000004
>>> print(s1.name,s2.name)
张三 李四
>>> s1.giveRaise(0.1)
** giveRaise:usetime=0.000009,alltime=0.000009
>>> s1.pay
55000.00000000001
>>> print(s1.lastName(),s2.lastName())
** lastName:usetime=0.000002,alltime=0.000002
** lastName:usetime=0.000004,alltime=0.000006
张 李

这篇关于python元类为类的全部方法添加装饰器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

python中的显式声明类型参数使用方式

《python中的显式声明类型参数使用方式》文章探讨了Python3.10+版本中类型注解的使用,指出FastAPI官方示例强调显式声明参数类型,通过|操作符替代Union/Optional,可提升代... 目录背景python函数显式声明的类型汇总基本类型集合类型Optional and Union(py

使用Java读取本地文件并转换为MultipartFile对象的方法

《使用Java读取本地文件并转换为MultipartFile对象的方法》在许多JavaWeb应用中,我们经常会遇到将本地文件上传至服务器或其他系统的需求,在这种场景下,MultipartFile对象非... 目录1. 基本需求2. 自定义 MultipartFile 类3. 实现代码4. 代码解析5. 自定

使用Python实现无损放大图片功能

《使用Python实现无损放大图片功能》本文介绍了如何使用Python的Pillow库进行无损图片放大,区分了JPEG和PNG格式在放大过程中的特点,并给出了示例代码,JPEG格式可能受压缩影响,需先... 目录一、什么是无损放大?二、实现方法步骤1:读取图片步骤2:无损放大图片步骤3:保存图片三、示php

Python文本相似度计算的方法大全

《Python文本相似度计算的方法大全》文本相似度是指两个文本在内容、结构或语义上的相近程度,通常用0到1之间的数值表示,0表示完全不同,1表示完全相同,本文将深入解析多种文本相似度计算方法,帮助您选... 目录前言什么是文本相似度?1. Levenshtein 距离(编辑距离)核心公式实现示例2. Jac

使用Python实现一个简易计算器的新手指南

《使用Python实现一个简易计算器的新手指南》计算器是编程入门的经典项目,它涵盖了变量、输入输出、条件判断等核心编程概念,通过这个小项目,可以快速掌握Python的基础语法,并为后续更复杂的项目打下... 目录准备工作基础概念解析分步实现计算器第一步:获取用户输入第二步:实现基本运算第三步:显示计算结果进

Python多线程实现大文件快速下载的代码实现

《Python多线程实现大文件快速下载的代码实现》在互联网时代,文件下载是日常操作之一,尤其是大文件,然而,网络条件不稳定或带宽有限时,下载速度会变得很慢,本文将介绍如何使用Python实现多线程下载... 目录引言一、多线程下载原理二、python实现多线程下载代码说明:三、实战案例四、注意事项五、总结引

Python利用PySpark和Kafka实现流处理引擎构建指南

《Python利用PySpark和Kafka实现流处理引擎构建指南》本文将深入解剖基于Python的实时处理黄金组合:Kafka(分布式消息队列)与PySpark(分布式计算引擎)的化学反应,并构建一... 目录引言:数据洪流时代的生存法则第一章 Kafka:数据世界的中央神经系统消息引擎核心设计哲学高吞吐

Python进阶之列表推导式的10个核心技巧

《Python进阶之列表推导式的10个核心技巧》在Python编程中,列表推导式(ListComprehension)是提升代码效率的瑞士军刀,本文将通过真实场景案例,揭示列表推导式的进阶用法,希望对... 目录一、基础语法重构:理解推导式的底层逻辑二、嵌套循环:破解多维数据处理难题三、条件表达式:实现分支

Java调用Python脚本实现HelloWorld的示例详解

《Java调用Python脚本实现HelloWorld的示例详解》作为程序员,我们经常会遇到需要在Java项目中调用Python脚本的场景,下面我们来看看如何从基础到进阶,一步步实现Java与Pyth... 目录一、环境准备二、基础调用:使用 Runtime.exec()2.1 实现步骤2.2 代码解析三、

C#高效实现Word文档内容查找与替换的6种方法

《C#高效实现Word文档内容查找与替换的6种方法》在日常文档处理工作中,尤其是面对大型Word文档时,手动查找、替换文本往往既耗时又容易出错,本文整理了C#查找与替换Word内容的6种方法,大家可以... 目录环境准备方法一:查找文本并替换为新文本方法二:使用正则表达式查找并替换文本方法三:将文本替换为图