J'essaie les annotations de type Python avec des classes de base abstraites pour écrire certaines interfaces. Existe-t-il un moyen d'annoter les types possibles de *args
et **kwargs
?
Par exemple, comment exprimerait-on que les arguments sensibles d'une fonction sont soit un int
ou deux int
s? type(args)
donne Tuple
donc ma supposition était d'annoter le type comme Union[Tuple[int, int], Tuple[int]]
, mais cela ne fonctionne pas.
from typing import Union, Tuple
def foo(*args: Union[Tuple[int, int], Tuple[int]]):
try:
i, j = args
return i + j
except ValueError:
assert len(args) == 1
i = args[0]
return i
# ok
print(foo((1,)))
print(foo((1, 2)))
# mypy does not like this
print(foo(1))
print(foo(1, 2))
Messages d'erreur de mypy:
t.py: note: In function "foo":
t.py:6: error: Unsupported operand types for + ("tuple" and "Union[Tuple[int, int], Tuple[int]]")
t.py: note: At top level:
t.py:12: error: Argument 1 to "foo" has incompatible type "int"; expected "Union[Tuple[int, int], Tuple[int]]"
t.py:14: error: Argument 1 to "foo" has incompatible type "int"; expected "Union[Tuple[int, int], Tuple[int]]"
t.py:15: error: Argument 1 to "foo" has incompatible type "int"; expected "Union[Tuple[int, int], Tuple[int]]"
t.py:15: error: Argument 2 to "foo" has incompatible type "int"; expected "Union[Tuple[int, int], Tuple[int]]"
Il est logique que mypy n'aime pas cela pour l'appel de fonction car il s'attend à ce qu'il y ait un tuple
dans l'appel lui-même. L'ajout après le déballage donne également une erreur de frappe que je ne comprends pas.
Comment annoter les types sensibles pour *args
et **kwargs
?
Optional
? Quelque chose a changé à propos de Python ou avez-vous changé d'avis? N'est-il toujours pas strictement nécessaire en raison duNone
défaut?