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

相关文章

Olingo分析和实践之ODataImpl详细分析(重要方法详解)

《Olingo分析和实践之ODataImpl详细分析(重要方法详解)》ODataImpl.java是ApacheOlingoOData框架的核心工厂类,负责创建序列化器、反序列化器和处理器等组件,... 目录概述主要职责类结构与继承关系核心功能分析1. 序列化器管理2. 反序列化器管理3. 处理器管理重要方

Python错误AttributeError: 'NoneType' object has no attribute问题的彻底解决方法

《Python错误AttributeError:NoneTypeobjecthasnoattribute问题的彻底解决方法》在Python项目开发和调试过程中,经常会碰到这样一个异常信息... 目录问题背景与概述错误解读:AttributeError: 'NoneType' object has no at

Python使用openpyxl读取Excel的操作详解

《Python使用openpyxl读取Excel的操作详解》本文介绍了使用Python的openpyxl库进行Excel文件的创建、读写、数据操作、工作簿与工作表管理,包括创建工作簿、加载工作簿、操作... 目录1 概述1.1 图示1.2 安装第三方库2 工作簿 workbook2.1 创建:Workboo

基于Python实现简易视频剪辑工具

《基于Python实现简易视频剪辑工具》这篇文章主要为大家详细介绍了如何用Python打造一个功能完备的简易视频剪辑工具,包括视频文件导入与格式转换,基础剪辑操作,音频处理等功能,感兴趣的小伙伴可以了... 目录一、技术选型与环境搭建二、核心功能模块实现1. 视频基础操作2. 音频处理3. 特效与转场三、高

Python实现中文文本处理与分析程序的示例详解

《Python实现中文文本处理与分析程序的示例详解》在当今信息爆炸的时代,文本数据的处理与分析成为了数据科学领域的重要课题,本文将使用Python开发一款基于Python的中文文本处理与分析程序,希望... 目录一、程序概述二、主要功能解析2.1 文件操作2.2 基础分析2.3 高级分析2.4 可视化2.5

一文解密Python进行监控进程的黑科技

《一文解密Python进行监控进程的黑科技》在计算机系统管理和应用性能优化中,监控进程的CPU、内存和IO使用率是非常重要的任务,下面我们就来讲讲如何Python写一个简单使用的监控进程的工具吧... 目录准备工作监控CPU使用率监控内存使用率监控IO使用率小工具代码整合在计算机系统管理和应用性能优化中,监

postgresql使用UUID函数的方法

《postgresql使用UUID函数的方法》本文给大家介绍postgresql使用UUID函数的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录PostgreSQL有两种生成uuid的方法。可以先通过sql查看是否已安装扩展函数,和可以安装的扩展函数

Python实现终端清屏的几种方式详解

《Python实现终端清屏的几种方式详解》在使用Python进行终端交互式编程时,我们经常需要清空当前终端屏幕的内容,本文为大家整理了几种常见的实现方法,有需要的小伙伴可以参考下... 目录方法一:使用 `os` 模块调用系统命令方法二:使用 `subprocess` 模块执行命令方法三:打印多个换行符模拟

Python实现MQTT通信的示例代码

《Python实现MQTT通信的示例代码》本文主要介绍了Python实现MQTT通信的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 安装paho-mqtt库‌2. 搭建MQTT代理服务器(Broker)‌‌3. pytho

Java中Arrays类和Collections类常用方法示例详解

《Java中Arrays类和Collections类常用方法示例详解》本文总结了Java中Arrays和Collections类的常用方法,涵盖数组填充、排序、搜索、复制、列表转换等操作,帮助开发者高... 目录Arrays.fill()相关用法Arrays.toString()Arrays.sort()A