这里是一个list
子类,它将其项目委托给itertools.compress
。
from itertools import compress
class WeirdList(list):
def __getitem__(self, item):
return compress(self, item)
l = WeirdList([1, 2, 3, 4])
print(*l[0, 1, 0, 1]) # 2 4
尽管我觉得我可以直接将compress
分配给__getitem__
,但上述方法效果很好。
class WeirdList(list):
__getitem__ = compress
l = WeirdList([1, 2, 3, 4])
print(*l[0, 1, 0, 1])
这引发了以下内容:
Traceback (most recent call last):
File "...", line 7, in <module> print(*l[0, 1, 0, 1])
TypeError: Required argument 'selectors' (pos 2) not found
我相信这会中断,因为compress
是一个类而不是一个函数,但是消息显示TypeError
是从调用compress
引发的。
__getitem__
协议在什么时候使用单个参数调用compress
?
答案 0 :(得分:3)
函数可以用作方法,因为它具有__get__
属性。 class compress
没有__get__
属性:
>>> compress.__get__
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: type object 'itertools.compress' has no attribute '__get__'
所以不能是方法。
使用__get__
方法调用属性时,将调用__get__
方法并返回其返回值,而不是属性本身的值。即l[0] == l.__getitem__(0) == l.__getitem__.__get__(l, type(l))(0)
,其中__get__
的返回值是一个已经将l
传递给函数的对象。
(如果您曾经想知道classmethod
和staticmethod
装饰器的作用,它们将使用不同的__get__
方法返回对象。)