Wypróbowuję adnotacje typu Pythona z abstrakcyjnymi klasami bazowymi, aby napisać niektóre interfejsy. Czy istnieje sposób na dodanie adnotacji do możliwych typów *args
i **kwargs
?
Na przykład, jak można by wyrazić, że sensowne argumenty funkcji to jeden int
lub dwa int
s? type(args)
daje Tuple
więc przypuszczam, że dodam adnotację jako Union[Tuple[int, int], Tuple[int]]
, ale to nie działa.
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))
Komunikaty o błędach od 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]]"
To ma sens, że mypy nie lubi tego dla wywołania funkcji, ponieważ oczekuje, że tuple
w samym wywołaniu będzie występować a . Dodatek po rozpakowaniu daje też błąd pisarski którego nie rozumiem.
W jaki sposób można opisać rozsądne typy dla *args
i **kwargs
?
źródło
Optional
? Czy coś się zmieniło w Pythonie, czy też zmieniłeś zdanie? Czy nadal nie jest to bezwzględnie konieczne ze względu naNone
domyślne ustawienie?Optional
adnotacja, gdy używaszNone
jako wartości domyślnej, utrudniła niektóre przypadki użycia i jest teraz usuwana z PEP.Optional
w przyszłości wymagane było wyraźne określenie.Callable
nie obsługuje żadnych wzmianek o podpowiedziach typu*args
lub**kwargs
kropce . Ta konkretna kwestia dotyczy zaznaczania wywołań, które akceptują określone argumenty oraz dowolną liczbę innych , a więc używaj*args: Any, **kwargs: Any
bardzo specyficznej wskazówki dotyczącej typu dla dwóch typów catch-all. W przypadkach, w których ustawisz*args
i / lub**kwargs
coś bardziej szczegółowego, możesz użyć plikuProtocol
.Właściwym sposobem na to jest użycie
@overload
Zauważ, że nie dodajesz
@overload
ani nie wpisujesz adnotacji do rzeczywistej implementacji, która musi być ostatnia.Będziesz potrzebować nowej wersji obu
typing
i mypy, aby uzyskać obsługę @overload poza plikami pośredniczącymi .Możesz również użyć tego do zróżnicowania zwracanego wyniku w sposób, który wyraźnie określa, które typy argumentów odpowiadają typowi zwracanemu. na przykład:
źródło
(type1)
vs(type1, type1)
jako mojego przykładu. Może(type1)
vs(type2, type1)
byłby lepszym przykładem i pokazuje, dlaczego podoba mi się ta odpowiedź. Pozwala to również na różne typy zwrotów. Jednak w szczególnym przypadku, gdy masz tylko jeden typ zwracania i wszystkie twoje*args
i*kwargs
są tego samego typu, technika w odpowiedzi Martjina ma więcej sensu, więc obie odpowiedzi są przydatne.*args
maksymalnej liczby argumentów (tutaj 2) jest nadal błędne .*args
koniecznie jest źle? Jeśli oczekiwane wywołania były(type1)
vs(type2, type1)
, liczba argumentów jest zmienna i nie ma odpowiedniej wartości domyślnej dla argumentu końcowego. Dlaczego ważne jest, aby było maksimum?*args
naprawdę istnieje dla zera lub więcej , nieograniczonych, jednorodnych argumentów lub dla „przekazania ich nietkniętych” argumentów typu catch-all. Masz jeden wymagany argument i jeden opcjonalny. Jest to zupełnie inne i zwykle jest obsługiwane przez nadanie drugiemu argumentowi domyślnej wartości wartownika, aby wykryć, że została pominięta.*args
, jeszcze lepszą odpowiedzią na pytanie jest to, że nie należy tego w ogóle robić.Jako krótki dodatek do poprzedniej odpowiedzi, jeśli starasz się używać mypy na Pythonie 2 pliki i trzeba używać komentarzy, aby dodać typy zamiast adnotacji, trzeba poprzedzić typy do
args
ikwargs
z*
i**
odpowiednio:Jest to traktowane przez mypy jako to samo, co poniżej, wersja Pythona 3.5
foo
:źródło