分类目录:《系统学习Python》总目录
抛开这些细节微妙性,Tracer
类装饰器示例最终仍然是依赖于__getattr__
来拦截对包装的和内嵌实例对象的获取。正如我们在前面见到的,我们真正需要完成的只是把实例创建调用移入一个类的内部,而不是把实例传递一个管理器函数。对于最初的非装饰器跟踪示例,我们可以用不同的方式编写实例创建:
class Spam:passfood = Wrapper(Spam())@Tracer
class Spam:passfood = 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] = objectreturn instances[aClass]bob = getInstance(Person('Bob'))
这对于我们前面所编写的跟踪器那样的函数装饰器也是成立的。我们可以直接把函数及其参数传递到负责分配调用的一个管理器中,而不是使用拦截随后调用的逻辑来装饰一个函数:
def func(x, y):passresult = tracer(func, (1, 2))@tracer
def func(x, y):passresult = func(1, 2)
像这样的管理器函数的方法把使用特殊语法的开销放到了调用上,而不是期待在函数和类定使用饰语法,但是它也允许我们在逐一调用的基础卜有选择地应用扩增。
参考文献:
[1] Mark Lutz. Python学习手册[M]. 机械工业出版社, 2018.