2

Python 3 引入了视图(参见这个问题)。它们也被向后移植到 Python 2.7。我目前正在对dictPython 2.7 应用程序进行子类化(尽管目标也是将其移植到 Python 3)。我想知道是否 - 以及如何 - 我可以将.viewitems()这些函数和类似函数子类化,使它们的行为与原始视图完全相同。

这是我的意图:我有一个这样的字典:

data = my_subclassed_dict
data["_internal"] = "internal_value"
data["key"] = "value"
list(data.keys()) == ["key"]

也就是说,我过滤了所有以 stat 开头的内容"_"。到目前为止,这工作得很好:对于迭代器,我只yield使用列表推导来过滤不需要的值。但是,这些项目dict不再有任何联系(这很好,感觉就像一个字典)。但是,这两种方式都行不通:

keys = data.viewkeys()
"key" in keys
del data["key"]
"key" not in keys # This is False !

最后一部分不起作用,因为没有引用原始键,所以 Python 不会注意到。

那么:有没有一种简单的方法来实现这一点(无需重新实现所有逻辑!)?

这更像是一个出于兴趣的问题,因为我认为它不会在我的场景中应用那么多。

4

2 回答 2

2

view对象基本上是“空”代理。他们指向原始字典。

不幸的是,当前的字典视图对象并不是真正可重用的。引用源代码

/* TODO(guido): The views objects are not complete:

 * support more set operations
 * support arbitrary mappings?
   - either these should be static or exported in dictobject.h
   - if public then they should probably be in builtins
*/

注意support arbitrary mappings条目;这些对象支持任意映射,我们也不能从 Python 代码创建新实例或子类化它们:

>>> type({}.viewkeys())({})
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: cannot create 'dict_keys' instances
>>> class MyView(type({}.viewkeys())): pass
... 
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: Error when calling the metaclass bases
    type 'dict_keys' is not an acceptable base type

您被迫创建自己的并实现视图对象支持的所有钩子:

class DictKeys(object):
    def __init__(self, parent):
        self.parent = parent

    def __len__(self):
        return len(self.parent)

    def __contains__(self, key):
        return key in self.parent

    def __iter__(self):
        return iter(self.parent)

等等

原始对象实现的方法是:

>>> dir({}.viewkeys())
['__and__', '__class__', '__contains__', '__delattr__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__iter__', '__le__', '__len__', '__lt__', '__ne__', '__new__', '__or__', '__rand__', '__reduce__', '__reduce_ex__', '__repr__', '__ror__', '__rsub__', '__rxor__', '__setattr__', '__sizeof__', '__str__', '__sub__', '__subclasshook__', '__xor__']

, __and__, __or__, __sub__, __xor__,和方法实现对 , 和 运算符的覆盖__rand__以提供集合操作。__ror____rsub____rxor__&|^

如果您在阅读 C 代码时相当安全,请查看视图对象实现以了解它们如何实现其方法。

于 2013-07-19T15:37:05.633 回答
0

考虑将任何以下划线开头的键存储在单独的字典中。您不必实现keys, iterkeys,viewitems等,但您可能需要重新实现许多其他方法。

于 2013-07-19T15:36:22.017 回答