首页 > 编程问答 >使用 mypy 时Python中的继承和多态性不起作用

使用 mypy 时Python中的继承和多态性不起作用

时间:2024-07-24 10:18:18浏览次数:25  
标签:python python-typing mypy

我正在寻找用 mypy 做一些标准的多态性,我以前从未使用过它,而且到目前为止它并不直观。

基类

class ContentPullOptions:
    pass


class Tool(Protocol):
    async def pull_content(self, opts: ContentPullOptions) -> str | Dict[str, Any]: ...

子类

class GoogleSearchOptions(ContentPullOptions):
    query: str
    sites: List[str]


class GoogleSearchTool(Tool):
    async def pull_content(
        self,
        opts: GoogleSearchOptions,
    ) -> str | Dict[str, Any]:

失败:

error: Argument 1 of "pull_content" is incompatible with supertype "Tool"; supertype defines the argument type as "ContentPullOptions" 

什么是像这样在 mypy 中通过类型检查进行基本继承的最可维护和干净的方法?

我尝试了自定义类型、强制转换等。但一切都感觉有点混乱和不清楚。

解决方案

也许这仍然违反了里氏原理

from typing import TypeVar, Protocol, Dict, Any, List, Callable

# Define T as contravariant
T_contra = TypeVar('T_contra', bound=ContentPullOptions, contravariant=True)

class ContentPullOptions:
    pass

class Tool(Protocol[T_contra]):
    async def pull_content(self, opts: T_contra) -> str | Dict[str, Any]: ...

class GoogleSearchOptions(ContentPullOptions):
    query: str
    sites: List[str]

class GoogleSearchTool:
    async def pull_content(
        self,
        opts: GoogleSearchOptions,
    ) -> str | Dict[str, Any]:
        # Implementation here
        pass

你遇到的 mypy 错误是因为你试图在 GoogleSearchTool 中专门化 pull_content 方法的参数类型,使其比 Tool 协议中定义的更窄。在静态类型语言中,这违反了里氏替换原则,该原则指出子类型应该能够替换其父类型而不会导致问题。

你可以使用以下方法之一在 mypy 中解决此问题并实现多态性:

1. 使用类型变量并保持 pull_content 的参数类型不变:

from typing import TypeVar, Protocol, Dict, Any, List

T = TypeVar('T', bound='ContentPullOptions')

class ContentPullOptions:
    pass

class Tool(Protocol[T]):
    async def pull_content(self, opts: T) -> str | Dict[str, Any]: ...

class GoogleSearchOptions(ContentPullOptions):
    query: str
    sites: List[str]

class GoogleSearchTool(Tool[GoogleSearchOptions]):
    async def pull_content(self, opts: GoogleSearchOptions) -> str | Dict[str, Any]:
        # Implementation here
        pass

此解决方案将类型变量 T 引入 Tool 协议,允许专门化 ContentPullOptions 的子类。 GoogleSearchTool 然后被声明为 Tool[GoogleSearchOptions] ,指示它使用 GoogleSearchOptions 作为 pull_content 的参数类型。

2. 使 Tool 中的参数类型更通用:

如果不需要在 Tool 级别指定具体类型,则可以使用 Any 类型:

from typing import Protocol, Dict, Any, List

class ContentPullOptions:
    pass

class Tool(Protocol):
    async def pull_content(self, opts: Any) -> str | Dict[str, Any]: ...

class GoogleSearchOptions(ContentPullOptions):
    query: str
    sites: List[str]

class GoogleSearchTool(Tool):
    async def pull_content(self, opts: GoogleSearchOptions) -> str | Dict[str, Any]:
        # Implementation here
        pass

此解决方案避免了对类型变量的需求,但在 Tool 级别丧失了类型安全性。

最佳方法取决于你的具体需求和你想在代码中实现的类型安全级别。如果需要在 Tool 级别保持类型安全性,第一个解决方案是更好的选择。但是,如果不需要这种级别的类型安全性,则可以使用第二个解决方案。

标签:python,python-typing,mypy
From: 78785680

相关文章

  • Python函数获取匹配和错误记录
    我有一个以下格式的json文件:[{"type":"BEGIN","id":"XYZ123"},{"type":"END","id":"XYZ123",},{"type":&......
  • python,替换标点符号但保持特殊单词完整的最佳方法
    我正在制作一个调制函数,它将采用带有特殊字符(@&*%)的关键字,并保持它们完整,同时从句子中删除所有其他标点符号。我设计了一个解决方案,但它非常庞大,而且可能比需要的更复杂。有没有一种方法可以以更简单的方式做到这一点。简而言之,我的代码匹配特殊单词的所有实例以查找跨度。然......
  • Python 检测 USB 设备 - IDLE 和 CMD 解释器之间的不同结果
    我正在尝试解决VDI解决方案中智能卡设备的USB重定向问题。我正在使用pyscard模块作为智能卡。对于进一步的上下文,主要问题是当浏览器插件调用用于处理智能卡的python脚本时,未检测到读卡器。关于问题,当我从CMD解释器运行此代码片段时,我收到空列表,表示系统上未找......
  • Python查找字符串的CRC32
    我尝试获取字符串数据类型变量的CRC32,但出现以下错误:>>>message='helloworld!'>>>importbinascii>>>binascii.crc32(message)Traceback(mostrecentcalllast):File"<stdin>",line1,in<module>TypeError:aby......
  • 使用python,如何创建重复的工作时间表
    这是我们公司的小组工作安排表。为三班制,2组日夜工作,1组休息。重复白天工作4天休息2天,然后再次夜间工作4天休息2天的时间表。我想使用python(pandas)自动安排在8月9日之后。抱歉英语不好,提前感谢您的帮助以下是使用Python和Pandas创建重复工作时间表的代码......
  • venv 已激活,但 pip 安装仍然默认进行,并且 python 在源代码中看不到该库
    在终端shell中的vscode中输入“whichpython”显示默认路径:C:\Users\erjan\AppData\Local\Programs\Python\Python311\python.exe(my_venv)但是(my_venv)意味着我的venv处于活动状态,我做了pipinstalltransformers,但下面的代码仍然显示错误-无法看到......
  • 在Python多处理中执行二进制信号量或互斥体以进行上下文切换操作
    我正在尝试自动化win应用程序和java应用程序之间的同步关系。我的标准是:启动win和jav应用程序在jav应用程序中执行命令等待jav应用程序的响应使用jav应用程序的响应到Windows应用程序作为输入。在jav应用程序中执行命令win应用程序......
  • 在spyder-python上随机出现的这些奇怪的亮点是什么
    在此处输入图像描述每次我单击此按钮或进行任何更改时,都会创建奇怪的突出显示,当我最小化功能时更是如此。有什么建议如何摆脱这些或可能的原因是什么?谢谢!我尝试更改外观首选项中的设置,但无法影响问题。很抱歉,我无法直接查看或与Spyder界面交互。我是一个AI......
  • 比较Python字典并找到缺失的元素
    我遇到了一个问题,我已经尝试了几天但没有得到任何结果。我想比较两个字典,在一个字典中有“赛前”足球比赛,在第二个字典中有“现场”足球比赛。我想将它们相互比较并打印它们(如果有)没有赛前比赛直播。示例1pre=[{"Home":"Genoa","Away":"In......
  • Python使用Visual Studio打印功能不显示输出
    任务:检查一个整数是正数还是负数。检查整数是否能被2整除。当输入0时,我需要退出循环并报告每个计数和总和。print函数没有显示任何输出。这是我从defmain()开始使用的代码defmain():countpositive=0countnegative=0count_divisible_by_2=0sump......