4

我正在努力寻找以下班级组织的“pythonic”方法:

我有一个基类,其属性在其构造函数中初始化,例如:

class Animal(object):
    def __init__(self, class_, species, is_domesticated):
        self.class_ = class_
        self.species = species
        self.is_domesticated = is_domesticated

然后,当我子类化时,我想“硬编码”这些属性中的一个或多个,如下所示:

class Mammal(Animal):
    def __init__(self, species, is_domesticated):
        Animal.__init__(self, 'Mammal', species, is_domesticated)

因此,哺乳动物的实例化如下:

monkey = Mammal('Primate', false)

问题是,我想使用 *args 以便在更改基类定义时不理会任何派生类。因此,哺乳动物的定义变为:

class Mammal(Animal):
    def __init__(self, *args):
        Animal.__init(self, *(args + (class_='Mammal',)))

哪个(不用说)看起来很可怕。一些提示将不胜感激=)

4

2 回答 2

6

如果您在基类中只有一组固定的参数,则无需担心可变参数。只需按照您在第一个示例中所做的操作即可。如果您希望能够向基类随机添加参数,但将它们添加为位置参数并且没有默认值,那么就没有希望了;您不能随意更改基类并期望所有派生类继续工作。

但是,有一个相当常见的中间情况,您可能拥有大量属性,这些属性的各种组合可以传递给层次结构中的任何类。您可能希望向基类添加新参数,但它们将具有默认值,因此派生类不需要显式了解它们;他们将优雅地降级为基类默认值。在这种情况下,使用**kwargs而不是*args.

class Animal(object):
    def __init__(self, **kwargs):
        self.class_ = kwargs['class_']
        self.species = kwargs['species']
        # etc.
class Mammal(Animal):
    def __init__(self, **kwargs):
        Animal.__init__(self, class_="Mammal", **kwargs)

这需要通过关键字传递的参数:

>>> Animal(class_='Fish', species='barracuda', is_domesticated=False)
4: <__main__.Animal object at 0x0177ABF0>
>>> Mammal(species="monkey", is_domesticated=False)
5: <__main__.Mammal object at 0x0177AFB0>

. . . 但如果它们有很多,那就更好了,因为如果你有 10 种不同的东西按位置传递,没有人会记得传递它们的顺序。这也意味着您可以轻松添加新参数;没有人必须知道在列表中的哪个位置放置新的,他们可以通过关键字将它们添加到任何地方。

在 Python 2 中,您必须像上面那样手动提取 kwargs。在 Python 3 中,您可以使用仅限关键字的参数来简化此操作。

于 2012-09-11T20:19:14.327 回答
0

好吧,你为什么不按你说的去做呢?提出Mammal.__init__()一个*args论点,然后使用它。这是代码:

class Animal(object):
    def __init__(self, class_, species, is_domesticated):
        self.class_ = class_
        self.species = species
        self.is_domesticated = is_domesticated
    def __str__(self):
        s_dom = "dom" if self.is_domesticated else "wild"
        return ("Animal(%s, %s, %s)" % (self.class_, self.species, s_dom))

class Mammal(Animal):
    def __init__(self, *args):
        Animal.__init__(self, 'Mammal', *args)

cat = Mammal("cat", True)
print(cat)

lion = Mammal("lion", False)
print(lion)

输出:

Animal(Mammal, cat, dom)
Animal(Mammal, lion, wild)
于 2012-09-11T20:43:29.657 回答