如何解决断言前的 Python 可选类型
我将 Python 类型提示和 mypy 与以下代码一起使用:
from typing import Optional
def compute(
use_wcog: bool = False,Xt: Optional[float] = None,Xd: Optional[float] = None,Xw: Optional[float] = None,Xs: Optional[float] = None,):
if use_wcog:
reqs = (Xt,Xd,Xw)
assert all([_ is not None for _ in reqs])
res = ((Xt**2 + Xw**2) / Xd)**2
else:
reqs = (Xs,Xd)
assert all([_ is not None for _ in reqs])
res = (Xs**2 / Xd)**2
return res
我收到以下错误: ** 不支持的操作数类型(浮点数和无)(mypy 错误)
正确的处理方式是什么?
解决方法
Mypy 不够聪明,无法根据 all
和列表推导来缩小类型。简单点:
if use_wcog:
assert Xt and Xd and Xw
res = ((Xt**2 + Xw**2) / Xd)**2
else:
assert Xs and Xd
res = (Xs**2 / Xd)**2
FWIW,将这些依赖项嵌入到函数体中使得输入在很大程度上无用 IMO——以一种在运行时会出错的方式调用这个函数很容易,而静态输入的全部意义在于产生使用错误可静态检测。鉴于该函数的两种实现在预期输入和返回结果方面具有完全不同的契约,我只是将其设为两个函数。
def compute_wcog(Xt: float,Xd: float,Xw: float) -> float:
return ((Xt**2 + Xw**2) / Xd)**2
def compute_no_wcog(Xd: float,Xs: float) -> float:
return (Xs**2 / Xd)**2
如果你真的需要一个具有多个接口的单一 compute
函数,你可以使用 @overload
装饰器:
from typing import Literal,Optional,overload
@overload
def compute(use_wcog: Literal[True],Xt: float,Xw: float,Xs: None) -> float: ...
@overload
def compute(use_wcog: Literal[False],Xt: None,Xw: None,Xs: float) -> float: ...
def compute(
use_wcog: bool = False,Xt: Optional[float] = None,Xd: Optional[float] = None,Xw: Optional[float] = None,Xs: Optional[float] = None,):
assert Xd is not None
if use_wcog:
assert Xt is not None and Xw is not None
res = ((Xt**2 + Xw**2) / Xd)**2
else:
assert Xs is not None
res = (Xs**2 / Xd)**2
return res
compute(True,1.0,None) # works
compute(False,None,1.0) # works
compute(True,1.0) # error: No overload variant of "compute" matches argument types...
但这显然是更多的代码,但收益并不多。 :)
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。