2

假设我有一个应用程序列表,描述为自定义对象。我决定对原始应用程序列表进行深度复制,这样我就可以拥有一个充当可用应用程序池的列表。为了分配应用程序,我随机选择池的一个子集,然后在原始列表中找到这些应用程序,以便我可以更新它们的信息。然后,从池中删除分配的应用程序。但是,由于某种原因,这些应用程序永远无法在原始列表中找到:

list1 = [App1,App2,App3,App4,App5]
pool = copy.deepcopy(list1)
num_apps = rand.randrange(0,5)
random.shuffle(pool)
selected_apps = copy.deepcopy(pool[:num_apps])
for app in selected_apps:
    locn = list1.index(app)
    print locn

上面的代码实际上会返回一个 ValueError ,表示找不到 selected_apps 中的项目。错误是因为我正在制作包含非 Python 原生对象的列表的深层副本,还是其他原因?

4

1 回答 1

0

使用list.index它时,使用比较运算符 ( ==) 来检查项目是否相同,然后使用__eq__方法,来自文档

用户定义的类默认有__eq__()__hash__()方法;与它们相比,所有对象都比较不相等(除了它们自己)x.__hash__()返回一个适当的值,这样就x == y意味着x is yhash(x) == hash(y)

您可以通过以下方式证明这一点:

>>> class A(object):
...     def __init__(self):
...         self.x = 0
... 
>>> a = A()
>>> a.x = 5
>>> b = a
>>> b == a
True
>>> c = copy.deepcopy(a)
>>> c == a
False

在 CPythonid()中返回对象的内存地址,因此__eq__如果 id 匹配则返回 true:

>>> id(a) 
140389541757224
>>> id(b)
140389541757224
>>> id(c)
140389541757336

所以你只需要实现__eq__方法和__ne__

比较运算符之间没有隐含的关系。的真理x==y并不意味着它x!=y是错误的。因此,在定义 时__eq__(),还应该定义__ne__()操作符的行为与预期一致。

>>> class A(object):
...     def __init__(self):
...         self.x = 0
...     def __eq__(self, o):
...         return self.x == o.x
...     def __ne__(self, o):
...         return self.x != o.x
... 
>>> a = A()
>>> a.x = 5
>>> b = a
>>> c = copy.deepcopy(a)
>>> d = A()
>>> d.x = 4
>>> a == b
True
>>> a == c
True
>>> a is b
True
>>> a is c
False
>>> a == d
False

还要注意这一点:

如果一个类没有定义一个__eq__()方法,它也不应该定义一个__hash__()操作;如果它定义__eq__()但未定义__hash__(),则它的实例将不能用作可散列集合中的项目

于 2015-05-13T20:16:52.833 回答