<p>由于您在类上定义了字段,实际的方法是键入hint字段。注意,您必须告诉<code>mypy</code>不要检查行本身。在</p>
<pre><code>class Person(PersonBase):
age: int = IntField() # type: ignore
</code></pre>
<p>这是最小的改变,但相当僵硬。在</p>
<hr/>
<p>通过使用带有伪签名的helper函数,可以创建自动键入的通用提示:</p>
^{pr2}$
<p>这就是<code>attrib</code>库提供其遗留提示的方式。这种风格允许隐藏注释的所有魔力/技巧。在</p>
<hr/>
<p>因为元类可以检查注释,所以不需要在字段中存储类型。您可以使用空的<code>Field</code>作为元数据,并为类型使用注释:</p>
<pre><code>from typing import Any
class Field(Any): # the (Any) part is only valid in a .pyi file!
"""Field description for Any type"""
class MetaPerson(type):
"""Metaclass that creates default class attributes based on fields"""
def __new__(mcs, name, bases, namespace, **kwds):
for name, value in namespace.copy().items():
if isinstance(value, Field):
# look up type from annotation
field_type = namespace['__annotations__'][name]
namespace[name] = field_type()
return super().__new__(mcs, name, bases, namespace, **kwds)
class Person(metaclass=MetaPerson):
age: int = Field()
</code></pre>
<p>这就是<code>attrib</code>提供其python3.6+属性的方式。它既通用又符合注释样式。请注意,这也可以用于常规基类而不是元类。在</p>
<pre><code>class BasePerson:
def __init__(self):
for name, value in type(self).__dict__.items():
if isinstance(value, Field):
field_type = self.__annotations__[name]
setattr(self, name, field_type())
class Person(BasePerson):
age: int = Field()
</code></pre>