how can I narrow dataclass annotations (i.e., how can I update type hints after handling default None in post_init)?

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

how can I narrow dataclass annotations (i.e., how can I update type hints after handling default None in post_init)?

问题

以下是代码的中文翻译部分:

  1. 我有一个数据类它可以接受一个关键字值或者如果未指定值则可以从其他属性推断出一个值
  2. import dataclasses
  3. @dataclasses.dataclass
  4. class RelatedValues:
  5. primary: float
  6. _: dataclasses.KW_ONLY
  7. secondary: float | None = None
  8. def __post_init__(self):
  9. if self.secondary is None:
  10. self.secondary = self.primary
  11. 这段代码有效但它让我在`.secondary`的类型提示中被困扰即使在`__post_init__`之后`.secondary`*绝对不可能*`None`
  12. `__post_init__`中对`self.secondary`进行`cast`操作不起作用但这个方法起作用
  13. NULL_FLOAT = float(int(uuid.uuid4())
  14. @dataclasses.dataclass
  15. class RelatedValues:
  16. primary: float
  17. _: dataclasses.KW_ONLY
  18. secondary: float = NULL_FLOAT
  19. def __post_init__(self):
  20. if self.secondary == NULL_FLOAT:
  21. self.secondary = self.primary
  22. 但这种方法感觉不太符合Python的风格
  23. 以下方法也有效
  24. @dataclasses.dataclass
  25. class RelatedValues:
  26. primary: float
  27. _: dataclasses.KW_ONLY
  28. _secondary: float | None = None
  29. def __post_init__(self):
  30. if self._secondary is None:
  31. self.secondary = self.primary
  32. else:
  33. self.secondary = self._secondary
  34. 或者这种方法也可以
  35. @dataclasses.dataclass
  36. class RelatedValues:
  37. primary: float
  38. _: dataclasses.KW_ONLY
  39. _secondary: float | None = None
  40. @property
  41. def secondary(self) -> float:
  42. if self._secondary is None:
  43. self.secondary = self.primary
  44. else:
  45. self.secondary = self._secondary
  46. 但后两种方法只是为了类型缩小而操纵我的kwargs这种做法有点不太对
  47. 我漏掉了什么
英文:

I have a dataclass that can take a keyword value, or, if no value is specified, infer a value from other attributes.

  1. import dataclasses
  2. @dataclasses.dataclass
  3. class RelatedValues:
  4. primary: float
  5. _: dataclasses.KW_ONLY
  6. secondary: float | None = None
  7. def __post_init__(self):
  8. if self.secondary is None:
  9. self.secondary = self.primary

This code works, but it leaves me stuck with float | None as the type hint for .secondary even though .secondary cannot possibly be None after __post_init__.

cast-ing self.secondary in __post_init__ doesn't work. This does:

  1. NULL_FLOAT = float(int(uuid.uuid4())
  2. @dataclasses.dataclass
  3. class RelatedValues:
  4. primary: float
  5. _: dataclasses.KW_ONLY
  6. secondary: float = NULL_FLOAT
  7. def __post_init__(self):
  8. if self.secondary == NULL_FLOAT:
  9. self.secondary = self.primary

But it feels distinctly non-Pythonic.

This also works:

  1. @dataclasses.dataclass
  2. class RelatedValues:
  3. primary: float
  4. _: dataclasses.KW_ONLY
  5. _secondary: float | None = None
  6. def __post_init__(self):
  7. if self._secondary is None:
  8. self.secondary = self.primary
  9. else:
  10. self.secondary = self._secondary

or this:

  1. @dataclasses.dataclass
  2. class RelatedValues:
  3. primary: float
  4. _: dataclasses.KW_ONLY
  5. _secondary: float | None = None
  6. @property
  7. def secondary(self) -> float:
  8. if self._secondary is None:
  9. self.secondary = self.primary
  10. else:
  11. self.secondary = self._secondary

But the latter two are just mangling my kwargs for the sake of type narrowing, which kind of feels wrong.

What am I missing?

答案1

得分: 1

如果 secondary 总是被赋予一个非 None 的值,那么在一开始就不应该将其类型定义为 float | None。实际上,只有作为 __init__ 方法的 参数 传递进来的 secondary_ 可能是 None

  1. from dataclasses import dataclass, field, InitVar
  2. @dataclass
  3. class RelatedValues:
  4. primary: float
  5. secondary: float = field(init=False)
  6. secondary_: InitVar[float|None] = None
  7. def __post_init__(self, secondary_):
  8. if secondary_ is None:
  9. secondary_ = self.primary
  10. self.secondary = secondary_

只有在你一开始不想提供自己的 __init__ 方法时才需要上述步骤。例如,

  1. @dataclass(init=False)
  2. class RelatedValues:
  3. primary: float
  4. secondary: float
  5. def __init__(self, primary: float, secondary: float = None):
  6. if secondary is None:
  7. secondary = primary
  8. self.primary = primary
  9. self.secondary = secondary
英文:

If secondary will always be assigned a non-None value, it should not be typed as float | None in the first place. It's only the argument to __init__, used to initialize self.secondary, that might be None.

  1. from dataclasses import dataclass, field, InitVar
  2. @dataclass
  3. class RelatedValues:
  4. primary: float
  5. secondary: float = field(init=False)
  6. secondary_: InitVar[float|None] = None
  7. def __post_init__(self, secondary_):
  8. if secondary_ is None:
  9. secondary_ = self.primary
  10. self.secondary = secondary_

These hoops are only necessary if you don't want to provide your own __init__ method in the first place. For example,

  1. @dataclass(init=False)
  2. class RelatedValues:
  3. primary: float
  4. secondary: float
  5. def __init__(self, primary: float, secondary: float = None):
  6. if secondary is None:
  7. secondary = primary
  8. self.primary = primary
  9. self.secondary = secondary

huangapple
  • 本文由 发表于 2023年2月7日 00:37:36
  • 转载请务必保留本文链接:https://go.coder-hub.com/75364117.html
匿名

发表评论

匿名网友

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

确定