TypeError在使用具有slots=True的数据类中使用super()时发生。

huangapple go评论107阅读模式
英文:

TypeError when using super() in a dataclass with slots=True

问题

这段代码中,当你在使用 dataclass 的时候,添加了 slots=True,就会导致 super() 的调用出现错误。为了修复这个问题,你可以在 super() 中使用类名而不是 self,像这样:

  1. data = super(B, self).get_data()

这会正确地调用父类的方法,而不会引发 TypeError。这是因为 slots=True 会创建一个新的类来代替原始的类,所以需要显式指定类名。

英文:

I have a dataclass with (kind of) a getter method.

This code works as expected:

  1. from dataclasses import dataclass
  2. @dataclass()
  3. class A:
  4. def get_data(self):
  5. # get some values from object's fields
  6. # do some calculations
  7. return "a calculated value"
  8. @dataclass()
  9. class B(A):
  10. def get_data(self):
  11. data = super().get_data()
  12. return data + " (modified)"
  13. b = B()
  14. print(b.get_data()) # a calculated value (modified)

However, if I add slots=True, I get a TypeError:

  1. from dataclasses import dataclass
  2. @dataclass(slots=True)
  3. class A:
  4. def get_data(self):
  5. return "a calculated value"
  6. @dataclass(slots=True)
  7. class B(A):
  8. def get_data(self):
  9. data = super().get_data()
  10. return data + " (modified)"
  11. b = B()
  12. print(b.get_data()) # TypeError: super(type, obj): obj must be an instance or subtype of type

The error vanishes if I use the old-style super(), contrary to pep-3135:

  1. from dataclasses import dataclass
  2. @dataclass(slots=True)
  3. class A:
  4. def get_data(self):
  5. return "a calculated value"
  6. @dataclass(slots=True)
  7. class B(A):
  8. def get_data(self):
  9. data = super(B, self).get_data()
  10. return data + " (modified)"
  11. b = B()
  12. print(b.get_data()) # a calculated value (modified)

Why does this happen and how to fix it the right way?

答案1

得分: 2

Sure, here's the translated content:

slots:如果设置为True(默认为False),将生成__slots__属性,并返回一个新的类,而不是原始类。如果类中已经定义了__slots__,则会引发TypeError

以此为例:

  1. @dataclass(slots=True)
  2. class Foo:
  3. pass

这意味着它的工作方式类似于:

  1. class Foo:
  2. pass

Foo = dataclass(slots=True)(Foo)

你定义了一个名为Foo的类,然后它被替换为一个不同的,被修改过的类。

现在,你的方法:

  1. def get_data(self):
  2. data = super().get_data()
  3. ...

这个super()是在原始的Foo类中编写的,它会假设它应该查找这个原始Foo类的父类;但它当前具有的实例实际上不是该类的实例,而是另一个经过修改的类的实例。

当你改用这种方式:

  1. data = super(Foo, self).get_data()

这会查找当前引用名称Foo的内容,这与此时self引用的内容相匹配。

英文:

> slots: If true (the default is False), __slots__ attribute will be generated and new class will be returned instead of the original one. If __slots__ is already defined in the class, then TypeError is raised.
>
> https://docs.python.org/3/library/dataclasses.html#dataclasses.dataclass

Taking this as an example:

  1. @dataclass(slots=True)
  2. class Foo:
  3. pass

This means this works something like this:

  1. class Foo:
  2. pass
  3. Foo = dataclass(slots=True)(Foo)

You define a class Foo, and then it gets replaced with a different, altered class.

Now, your method:

  1. def get_data(self):
  2. data = super().get_data()
  3. ...

This super() was written in the original Foo class and will assume it's supposed to look up the parent of this original Foo class; but the instance it currently has is not actually an instance of that class, it's an instance of the other, altered class.

When you do this instead:

  1. data = super(Foo, self).get_data()

This looks up what currently refers to the name Foo, which again matches what self also refers to at this moment.

huangapple
  • 本文由 发表于 2023年8月4日 21:29:54
  • 转载请务必保留本文链接:https://go.coder-hub.com/76836403.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定