gpt4 book ai didi

python - __getitem__、__setitem__ 如何处理切片?

转载 作者:太空宇宙 更新时间:2023-11-03 15:06:01 24 4
gpt4 key购买 nike

我正在运行 Python 2.7.10。

我需要拦截列表中的更改。 “更改”是指任何在浅层意义上修改列表的内容(如果列表由相同顺序的相同对象组成,则列表不会更改,而不管这些对象的状态如何;否则,它是)。我不需要了解列表如何发生了变化,只要它发生了变化即可。所以我只是确保我可以检测到它,然后让基本方法完成它的工作。这是我的测试程序:

class List(list):
def __init__(self, data):
list.__init__(self, data)
print '__init__(', data, '):', self

def __getitem__(self, key):
print 'calling __getitem__(', self, ',', key, ')',
r = list.__getitem__(self, key)
print '-->', r
return r

def __setitem__(self, key, data):
print 'before __setitem__:', self
list.__setitem__(self, key, data)
print 'after __setitem__(', key, ',', data, '):', self

def __delitem__(self, key):
print 'before __delitem__:', self
list.__delitem__(self, key)
print 'after __delitem__(', key, '):', self

l = List([0,1,2,3,4,5,6,7]) #1
x = l[5] #2
l[3] = 33 #3
x = l[3:7] #4
del l[3] #5
l[0:4]=[55,66,77,88] #6
l.append(8) #7

案例 #1、#2、#3 和 #5 的工作符合我的预期; #4、#6 和#7 没有。程序打印:

__init__( [0, 1, 2, 3, 4, 5, 6, 7] ): [0, 1, 2, 3, 4, 5, 6, 7]
calling __getitem__( [0, 1, 2, 3, 4, 5, 6, 7] , 5 ) --> 5
before __setitem__: [0, 1, 2, 3, 4, 5, 6, 7]
after __setitem__( 3 , 33 ): [0, 1, 2, 33, 4, 5, 6, 7]
before __delitem__: [0, 1, 2, 33, 4, 5, 6, 7]
after __delitem__( 3 ): [0, 1, 2, 4, 5, 6, 7]

我对#7 并不感到非常惊讶:append 可能是以一种特别的方式实现的。但是对于#4 和#6 我很困惑。 __getitem__ 文档说:“调用以实现 self[key] 的评估。对于序列类型,接受的键应该是整数和 slice 对象。”(我的重点)。对于 __setitem__:“与 __getitem__()” 相同的注释,我认为这意味着 key 可以也是一片。

我的推理有什么问题?如有必要,我准备重写每个列表修改方法(附加、扩展、插入、弹出等),但是应该重写什么以捕获类似 #6 的内容?

我知道 __setslice__ 等的存在。但这些方法自 2.0 以来已被弃用 ...

嗯。我再次阅读了有关 __getslice____setslice__ 等的文档,我发现了这个令人毛骨悚然的声明:

“(但是,CPython 中的内置类型目前仍然实现了 __getslice__()。因此,在实现切片时,您必须在派生类中覆盖它。)”

这是解释吗?这是在说“好吧,方法已被弃用,但为了在 2.7.10 中实现与 2.0 中相同的功能,您仍然必须覆盖它们”? las,那您为什么不赞成使用它们? future 将如何运作?是否有一个“列表”类——我不知道——我可以扩展并且不会带来这种不便?我真正需要重写什么以确保捕获每个列表修改操作?

最佳答案

问题是您要对内置函数进行子类化,因此必须处理一些问题。在深入探讨该问题之前,我将直接进入“现代”方式:

How will things work in the future? Is there a "list" class - that I am not aware of - that I could extend and would not present this inconvenience?

是的,有标准库 Abstract Base Classes .您可以通过使用 ABC 来避免由子类化内置 list 引起的丑陋的并发症。对于类似列表的东西,尝试子类化 MutableSequence :

from collections import MutableSequence

class MyList(MutableSequence):
...

现在你应该只需要处理 __getitem__ 和 friend 的切片行为。


如果您想继续继承内置的 list,请继续阅读...

您的猜测是正确的,您将需要覆盖 __getslice____setslice__language reference explains why你已经看到了:

However, built-in types in CPython currently still implement __getslice__(). Therefore, you have to override it in derived classes when implementing slicing.

请注意,l[3:7] 将 Hook 到 __getslice__,而其他等效的 l[3:7:] 将 Hook 进入 __getitem__,所以你必须处理在这两种方法中接收切片的可能性......呻吟!

关于python - __getitem__、__setitem__ 如何处理切片?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32772755/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com