<p>{t>在创建一个包装好的对象时,{cd1>不创建一个特殊的方法。像方法一样,每个访问都会创建一个新对象,在本例中是一个新的函数对象。在</p>
<pre><code>class A:
def __init__(self):
self.data = 'instance'
@combomethod
def foo(param):
if isinstance(param, A):
print("This is an " + param.data + " method.")
elif param is A:
print("This is a class method.")
>>> a = A()
>>> A.foo
<function foo at 0x00CFE810>
>>> a.foo
<function foo at 0x00CFE858>
>>> A.foo()
This is a class method.
>>> a.foo()
This is an instance method.
</code></pre>
<p>每次访问都是新的:</p>
^{pr2}$
<p><code>foo</code>实际上是{<cd3>}的伪装:</p>
<pre><code>>>> A.foo.__code__.co_name
'_wrapper'
</code></pre>
<p>当从类调用时,闭包具有obj==None(注意这里的“self”指的是combMethod,它引用了<code>self.method</code>中的原始函数对象):</p>
<pre><code>>>> print(*zip(A.foo.__code__.co_freevars, A.foo.__closure__), sep='\n')
('obj', <cell at 0x011983F0: NoneType object at 0x1E1DF8F4>)
('self', <cell at 0x01198530: combomethod object at 0x00D29630>)
('objtype', <cell at 0x00D29D10: type object at 0x01196858>)
</code></pre>
<p>当作为实例的属性调用时,obj是实例:</p>
<pre><code>>>> print(*zip(a.foo.__code__.co_freevars, a.foo.__closure__), sep='\n')
('obj', <cell at 0x01198570: A object at 0x00D29FD0>)
('self', <cell at 0x01198530: combomethod object at 0x00D29630>)
('objtype', <cell at 0x00D29D10: type object at 0x01196858>)
</code></pre>
<p>以下是存储在combomethod中的原始函数:</p>
<pre><code>>>> A.foo.__closure__[1].cell_contents.method
<function foo at 0x00D1CB70>
>>> A.foo.__closure__[1].cell_contents.method.__code__.co_name
'foo'
</code></pre>
<p><code>_wrapper</code>执行<code>self.method</code>,将类或实例作为给定obj值的第一个参数:</p>
<pre><code>if obj is not None:
return self.method(obj, *args, **kwargs)
else:
return self.method(objtype, *args, **kwargs)
</code></pre>