gpt4 book ai didi

python - __repr__ 装饰器在子类上失败

转载 作者:太空宇宙 更新时间:2023-11-04 08:35:49 28 4
gpt4 key购买 nike

背景

我编写了一个修饰函数来修改给定类的 __repr__,这样当调用一个类实例时,它的所有属性都会打印给用户。当在下面示例中的 Container 类中使用时,装饰器 __repr__dec 的行为符合预期。

输入

def __repr__wrapper(self):
"""Show all attributes."""
return "Attributes: "+", ".join(list(self.__dict__.keys()))

def __repr__dec(func):
"""Replaces the __repr__ function of a class with __repr__wrapper"""
def call(*args, **kwargs):
func.__repr__ = __repr__wrapper
result = func(*args, **kwargs)
return result
return call


@__repr__dec
class Container(object):
def __init__(self, *args, **kwargs):
self.metadata = args[0]
for k,v in kwargs.items():
self.__dict__[k] = v

occ = Container(42, how="now")
occ

输出

Attributes: metadata, how

然而,当尝试子类化 Container 时,我收到一条 TypeError 消息:

输入

class Handle(Container):
def __init__(self, *args, **kwargs):
Container.__init__(self, *args, **kwargs)

han = Handle(42)

输出

TypeError                                 Traceback (most recent call last)
<ipython-input-17-b4c252411c1f> in <module>()
----> 1 class Handle(Container):
2 def __init__(self, *args, **kwargs):
3 Container.__init__(self, *args, **kwargs)
4
5 han = Handle(42)

TypeError: function() argument 1 must be code, not str

问题

为什么在使用 __repr__dec 函数时子类化 Conatainer 会失败?有可能解决这个问题吗?

最佳答案

问题是您的装饰器使 Container 成为一个函数而不再是一个类。您可以非常简单地控制它:

>>> type(Container)
<class 'function'>

这是因为你对装饰器的使用结束于:

  • 声明一个未修饰的类

    class Container:
    ...
  • 在上面使用装饰器:

    Container = __repr__dec(Container)

由于 __repr__dec 返回一个函数,您确实将 Container 更改为能够返回具有预期 __repr__ 成员的对象的函数,但它是不再是一个类。

如果您希望以后能够对其进行子类化,您的装饰器必须返回一个类:

def repr_dec(cls):
cls.__repr__ = __repr__wrapper
return cls

然后一切正常:

>>> Container
<class '__main__.Container'>
>>> occ=Container(42, how="now")
>>> occ
Attributes: metadata, how

并且您可以成功地对其进行子类化:

>>> class Handle(Container):
def __init__(self, *args, **kwargs):
Container.__init__(self, *args, **kwargs)

>>> han = Handle(42, foo="now")
>>> han
Attributes: metadata, foo

Handle 类从其父类继承了 __repr__ 方法。

关于python - __repr__ 装饰器在子类上失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49197081/

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