gpt4 book ai didi

python - 子类化 dict : should dict. __init__() 被调用?

转载 作者:IT老高 更新时间:2023-10-28 22:00:55 29 4
gpt4 key购买 nike

这是一个双重问题,一个理论部分,一个实践部分:

当子类化dict时:

class ImageDB(dict):
def __init__(self, directory):
dict.__init__(self) # Necessary??
...

是否应该调用 dict.__init__(self),作为“安全”措施(例如,如果有一些重要的实现细节很重要)?如果 not 调用 dict.__init__() 是否存在代码与 Python 的 future 版本中断的风险?我在这里寻找做一件事或另一件事的根本原因(实际上,调用 dict.__init__() 是安全的)。

我的猜测是,当调用 ImageDB.__init__(self, directory) 时,self 已经是一个新的空 dict 对象,因此不需要调用 dict.__init__ (起初我确实希望 dict 为空)。这是正确的吗?

编辑:

上述基本问题背后更实际的问题如下。我正在考虑将 dict 子类化,因为我会经常使用 db[...] 语法(而不是一直使用 db.contents[...]);对象的唯一数据(属性)确实是一个字典。我想向数据库中添加一些方法(例如 get_image_by_name()get_image_by_code()),并且只覆盖 __init__() ,因为图像数据库是由包含它的目录定义的。

总而言之,(实际)问题可能是:对于表现得像字典的东西,什么是好的实现,除了它的初始化不同(它只需要一个目录名称),以及它有额外的方法吗?

许多答案中都提到了“工厂”。所以我想这一切都归结为:你是继承 dict,重写 __init__() 并添加方法,还是编写一个返回 dict 的(工厂)函数,你添加方法?我倾向于第一种方案,因为工厂函数返回一个对象,其类型并不表示它有额外的语义和方法,但你怎么看?

编辑 2:

我从每个人的回答中得知,当新类“不是字典”时,将 dict 子类化不是一个好主意,特别是当它的 __init__ 方法不能采用与 dict 相同的参数时__init__ (上面的“实际问题”中就是这种情况)。换句话说,如果我理解正确的话,共识似乎是:当您子类化时,所有方法(包括初始化)必须具有与基类方法相同的签名。这允许 isinstance(subclass_instance, dict) 保证 subclass_instance.__init__() 可以像 dict.__init__() 一样使用。

接下来又出现了一个实际的问题:除了初始化方法之外,和dict一样的类应该如何实现呢?没有子类化?这需要一些麻烦的样板代码,不是吗?

最佳答案

你应该在子类化时调用 dict.__init__(self);实际上,您不知道 dict 中究竟发生了什么(因为它是内置的),并且可能因版本和实现而异。不调用它可能会导致不当行为,因为您无法知道 dict 在哪里保存其内部数据结构。

顺便说一句,你没有告诉我们你想要做什么;如果您想要一个具有 dict(映射)行为的类,并且您实际上并不需要 dict(例如,在您的软件中的任何地方都没有执行 isinstance(x, dict) 的代码,因为它应该是) ,如果你在 python <= 2.5 或 collections.MutableMapping 上,你可能最好使用 UserDict.UserDictUserDict.DictMixin > 如果你在 python >= 2.6 上。这些将为您的类(class)提供出色的听写行为。

编辑:我在另一条评论中读到你没有覆盖任何 dict 的方法!那么子类化就没有意义了,不要这样做。

def createImageDb(directory):
d = {}
# do something to fill in the dict
return d

编辑 2:您想从 dict 继承以添加新方法,但您不需要覆盖任何方法。一个不错的选择可能是:

class MyContainer(dict):
def newmethod1(self, args):
pass

def newmethod2(self, args2):
pass


def createImageDb(directory):
d = MyContainer()
# fill the container
return d

顺便问一下:你要添加什么方法?你确定你正在创建一个好的抽象吗?也许你最好使用一个定义你需要的方法的类,并在它内部使用一个“普通”的字典。

工厂功能: http://en.wikipedia.org/wiki/Factory_method_pattern

这只是将实例的构造委托(delegate)给函数而不是覆盖/更改其构造函数的一种方式。

关于python - 子类化 dict : should dict. __init__() 被调用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2033150/

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