gpt4 book ai didi

python - 如何类型提示 Python 函数返回从父类(super class)派生的任何类的实例?

转载 作者:行者123 更新时间:2023-11-30 22:05:21 24 4
gpt4 key购买 nike

我有一堆 Django 模板包含标签,它们将数据库对象的特定实例或字符串/整数作为参数,它被解释为该数据库对象的主键。例如...

{% render_product product=obj %}
{% render_product product=42 %}
{% render_product product="42" %}

...一切都工作正常并且执行明显的操作:它们使用特定的 Product 实例呈现模板片段,如果需要,可以通过主键从数据库中获取它。这是 Product 和类似类的定义方式:

class Product(models.Model):
# standard django model definition goes here

以下是此类包含标记中通常会发生的情况:

@register.inclusion_tag("render_product.html")
def render_product(product: Union[Product, str, int] = None) -> dict:
_product = None
if isinstance(product, Product):
_product = product
elif isinstance(product, str) or isinstance(product, int):
try:
_product = Product.objects.get(pk=product)
except (Product.DoesNotExist, ValueError):
pass
return {"product": _product}

由于我在数十个包含标签中出现了相同的模式,因此我尝试将其重构,以便得到类似的内容:

@register.inclusion_tag("render_product.html")
def render_product(product: Union[Product, str, int] = None) -> dict:
_product = fetch_object(Product, product)
return {"product": _product}

这是 fetch_object 代码:

def fetch_object(cls: Type[Model] = None, obj: Union[Model, str, int] = None):
if isinstance(obj, cls):
return obj
elif isinstance(obj, str) or isinstance(obj, int):
try:
return cls.objects.get(pk=obj)
except (cls.DoesNotExist, ValueError):
pass
return None

我的问题是:我不知道如何指定该函数的返回类型。基本上它应该类似于“任何类的实例,从 Model 或 None 派生”。但如果我尝试类似的事情......

def fetch_object(
cls: Type[Model] = None, obj: Union[Model, str, int] = None
) -> Union[Model, None]:

...如果我访问所获取对象上的方法(特定于产品,而不是特定于模型),PyCharm 会提示“未解析的属性引用”。

我正在尝试在我的Python代码中使用越来越多的类型提示,因为它已经救了我几次,但这是其中一种情况,我不知道正确的做法是什么确实如此,但我的 google-fu 却让我失望了。

fetch_object 的正确类型提示是什么?

最佳答案

您要做的就是使您的 fetch_object 函数成为 generic function .

也就是说,而不只是说您的函数接受任何 Type[Model],使用类型变量准确捕获您接受的模型类型,并指定输出的确切类型。例如:

from typing import TypeVar

# The bound states that T can be bound to Model or any subclass of Model.
# If the bound keyword argument is omitted, we assume the bound is 'object'.
T = TypeVar('T', bound=Model)

def fetch_object(cls: Type[T] = None, obj: Union[T, str, int] = None) -> Optional[T]:
if isinstance(obj, cls):
return obj
elif isinstance(obj, str) or isinstance(obj, int):
try:
return cls.objects.get(pk=obj)
except (cls.DoesNotExist, ValueError):
pass
return None

关于风格约定的一个小注释:为了简洁起见,我选择在这里将类型变量命名为 T。另一个常见的约定是将类型变量命名为 _TModel_ModelT。也就是说,使用下划线使变量私有(private),并使用更长的名称以提高可读性。

关于python - 如何类型提示 Python 函数返回从父类(super class)派生的任何类的实例?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53049652/

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