首页 > 编程问答 >计数调用装饰器 - 为什么我将函数属性重置回 0?

计数调用装饰器 - 为什么我将函数属性重置回 0?

时间:2024-07-24 13:29:46浏览次数:8  
标签:python python-decorators

下面的代码计算了装饰函数 func 被调用的次数:

from functools import wraps

def counting_calls(func):
    @wraps(func)
    def inner(*args, **kwargs):
        inner.call_count += 1
        return func(*args, **kwargs)
    inner.call_count = 0
    return inner

对于这个测试:

@counting_calls
def add(a: int, b: int) -> int:
    '''return sum of 2 ints'''
    return a + b

print(add(10, b=20))
print(add(30, 5))
print(add(3, 5))
print(add(4, 5))
print('num calls =', add.call_count)
print(add(11, 5))
print('num calls =', add.call_count)

为什么我们要重置inner.call_count = 0?它实际上不会将 call_count 始终重置为 0 并且不保存总调用次数吗?


你对 inner.call_count = 0 行的作用理解得非常正确。你绝对正确,这行代码会将 call_count 重置为 0,它不会保存装饰函数的总调用次数。

原因是我们希望每次调用 inner 时, call_count 都会递增。如果我们不将 inner.call_count 初始化为 0,它在第一次调用时将是未定义的,并且会出现 UnboundLocalError 错误。

下面是如何修复代码以正确保存总调用次数的方法:

from functools import wraps

def counting_calls(func):
    @wraps(func)
    def inner(*args, **kwargs):
        inner.call_count += 1
        return func(*args, **kwargs)
    inner.call_count = 0  # 在这里初始化计数器
    return inner

通过这种更改, call_count 将在装饰器定义时初始化一次,然后每次调用装饰函数时递增。

以下是更新后的代码及其工作原理:

from functools import wraps

def counting_calls(func):
    @wraps(func)
    def inner(*args, **kwargs):
        inner.call_count += 1
        return func(*args, **kwargs)
    inner.call_count = 0
    return inner

@counting_calls
def add(a: int, b: int) -> int:
    '''return sum of 2 ints'''
    return a + b

print(add(10, b=20))
print(add(30, 5))
print(add(3, 5))
print(add(4, 5))
print('num calls =', add.call_count)
print(add(11, 5))
print('num calls =', add.call_count)

这现在将输出:

30
35
8
9
num calls = 4
16
num calls = 5

如所见, call_count 现在正确跟踪 add 函数的总调用次数。

标签:python,python-decorators
From: 78786496

相关文章

  • Python 类型提示:自引用类型检查
    我认为静态类型确实会让我在Python中的生活更轻松,考虑到我通常使用C语言工作。我创建了一个像这样的类:classnode(object):"""properties,constructor,etc."""defadd_outneighbor(self,neighbor:node)->None:"""d......
  • Python用shp文件裁剪多个遥感影像的方法
      本文介绍基于Python中ArcPy模块,基于矢量数据范围,对大量栅格遥感影像加以批量裁剪掩膜的方法。  首先,话不多说,本文所需要的代码如下所示。#-*-coding:utf-8-*-"""CreatedonTueDec1320:07:482022@author:fkxxgis"""importarcpyfromarcpy.saimport*......
  • 以编程方式删除 Python 源中的类型提示
    我有一些为Python3.5编写的源代码,我想使其在Python3.4下可执行。我使用的3.5中唯一在3.4中不可用的功能是类型提示,因此我想编写一个脚本来完全删除这些功能。乍一看这似乎很容易,我决定编写一些正则表达式这样做,但后来我想到了一些边缘情况,我不确定如何解决像这样的......
  • Python 类型暗示​​一个充满 myclass 对象的双端队列
    使用Python3.6或更高版本,我想输入提示一个返回MyClass对象的函数myfunc我如何提示myqueue是一个deque|||充满MyClass对象?objects?fromcollectionsimportdequeglobal_queue=deque()classMyClass:passdefmyfunc(m......
  • python之名称空间和作用域(关键字:global和nonlocal的使用)
    文章目录前言1、名称空间和作用域1.1引言1.2名称空间1.2.1内置名称空间1.2.2全局名称空间1.2.3局部名称空间1.2.4名称空间的产生和销毁顺序1.3作用域1.3.1全局作用域1.3.2局部作用域1.3.3名字的查找顺序1.4关键字:global1.5关键字:nonlocal前言本篇文章......
  • 用于 isinstance() 检查的 dict_keys 的显式 python3 类型是什么?
    在Python3中,我应该使用什么类型来检查字典键是否属于它?>>>d={1:2}>>>type(d.keys())<class'dict_keys'>所以我很自然地尝试了这个:>>>isinstance(d.keys(),dict_keys)Traceback(mostrecentcalllast):File"<stdin>",......
  • 初学Python时需要认识清楚的几个概念:对象、函数、圆括号给、点取、方括号取
    这是我在自学Python的过程中自己挑选和提炼出来的几个重要的概念,之所以特意介绍这些概念,其中包含了我自己的思维方式和我对Python设计理念的认识,有其独特性和局限性。我希望这篇文章能够给喜爱Python的朋友们带来一些启发。1、对象(Object)对象是Python编程的基本单元。就像音是......
  • 如何接受文件或路径作为python中方法的参数
    我正在尝试编写一个方法,该方法将接受打开的文件myFile=open("myFile.txt")obj.writeTo(myFile)myFile.close()或带有路径的字符串obj.writeTo("myFile.txt")该方法的实现如下:defwriteTo(self,hessianFile):ifisinstance(hessianFile,file):pr......
  • Python,“pathlib.Path.open()”方法和内置函数“open()”不返回“typing.IO”的实例
    我读过一些其他答案,它们似乎与现实生活中发生的事情相矛盾。尝试以下操作:fromtypingimportIOfrompathlibimportPathexample_path=Path(r"D:\Example.txt")withexample_path.open("r")asf:print(isinstance(f,IO))withopen(example_path)a......
  • 【Dison夏令营 Day 28】用 Python 创建恐龙游戏
    谁没有玩过谷歌著名的“恐龙游戏”?也许每个人都玩过这个游戏。今天,在这篇文章中,我们将帮助你用Python开发一个恐龙游戏。本教程将深入讲解每一行代码,并提供参考资料。我们将尽力让读者详细、透彻地理解这个项目。Python版恐龙游戏的任务记录包括图片文档和Python资料......