系统学习Python——装饰器:类装饰器-[装饰器与管理器函数]

本文讨论了在Python中如何使用Tracer类装饰器实现功能扩展,通过类装饰器和管理器函数的方式,将特殊语法需求从实例创建移到类定义或函数调用中,展示了不同策略的灵活性和适用场景。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

分类目录:《系统学习Python》总目录


抛开这些细节微妙性,Tracer类装饰器示例最终仍然是依赖于__getattr__来拦截对包装的和内嵌实例对象的获取。正如我们在前面见到的,我们真正需要完成的只是把实例创建调用移入一个类的内部,而不是把实例传递一个管理器函数。对于最初的非装饰器跟踪示例,我们可以用不同的方式编写实例创建:

class Spam:
    pass

food = Wrapper(Spam())

@Tracer
class Spam:
    pass

food = Spam()

基本上,类装饰器把特殊语法要求从实例创建调用迁移到了类语句自身。对于前面文章中的单例类示例来说也是如此,我们直接将类及其构造参数传递到了一个管理器函数中,而不是装饰一个类并使用常规的实例创建调用:

instances = {}
def getInstance(aClass, *args, **kargs):
    if aClass not in instances:
        instances[aClass] = aClass(*args, **kargs)
    return instances[aClass]

bob = getInstance(Person, 'Bob')

另一种方案是,我们可以使用Python的内省工具从一个已经创建好的实例中获取类(假设创建一个初始化实例是可以接受的):

instances = {}
def getInstance(object):
    aClass = object.__class__
    if aClass not in instances:
        instances[aClass] = object
    return instances[aClass]

bob = getInstance(Person('Bob'))

这对于我们前面所编写的跟踪器那样的函数装饰器也是成立的。我们可以直接把函数及其参数传递到负责分配调用的一个管理器中,而不是使用拦截随后调用的逻辑来装饰一个函数:

def func(x, y):
    pass

result = tracer(func, (1, 2))

@tracer
def func(x, y):
    pass

result = func(1, 2)

像这样的管理器函数的方法把使用特殊语法的开销放到了调用上,而不是期待在函数和类定使用饰语法,但是它也允许我们在逐一调用的基础卜有选择地应用扩增。

参考文献:
[1] Mark Lutz. Python学习手册[M]. 机械工业出版社, 2018.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

von Neumann

您的赞赏是我创作最大的动力~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值