* args 및 ** kwargs에 대한 유형 주석


158

일부 기본 인터페이스를 작성하기 위해 추상 기본 클래스를 사용하여 Python의 유형 주석을 시도하고 있습니다. 의 가능한 유형의 주석을 할 수있는 방법이 있나요 *args**kwargs?

예를 들어, 함수에 대한 합리적인 인수가 하나 int또는 두 개 라는 것을 어떻게 표현할 수 int있습니까? type(args)제공 Tuple내 생각 엔이 같은 유형에 주석을했다, 그래서 Union[Tuple[int, int], Tuple[int]],하지만이 작동하지 않습니다.

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))

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]]"

mypy는 tuple호출 자체에 있을 것으로 기대하기 때문에 함수 호출에 대해 이것을 좋아하지 않는 것이 좋습니다 . 포장 풀기 후 추가하면 이해할 수없는 입력 오류가 발생합니다.

어떻게 하나의 분별 유형을 주석 않습니다 *args**kwargs?

답변:


167

가변 위치 인수 ( *args) 및 가변 키워드 인수 ( **kw)의 경우 이러한 인수 중 하나에 대한 예상 값만 지정하면됩니다 .

유형 힌트 PEP 의 임의 인수 목록 및 기본 인수 값 섹션 에서 다음을 수행하십시오 .

임의의 인수 목록은 형식 주석을 달 수 있으므로 정의는 다음과 같습니다.

def foo(*args: str, **kwds: int): ...

예를 들어, 다음은 모두 유효한 유형의 인수를 가진 함수 호출을 나타냅니다.

foo('a', 'b', 'c')
foo(x=1, y=2)
foo('', z=0)

따라서 다음과 같이 방법을 지정하고 싶습니다.

def foo(*args: int):

그러나 함수가 하나 또는 두 개의 정수 값만 허용하는 경우 전혀 사용하지 말고 *args명시적인 위치 인수와 두 번째 키워드 인수를 사용하십시오.

def foo(first: int, second: Optional[int] = None):

이제 함수는 실제로 하나 또는 두 개의 인수로 제한되며, 지정된 경우 둘 다 정수 여야합니다. *args 항상 0 이상을 의미하며 유형 힌트로보다 구체적인 범위로 제한 할 수 없습니다.


1
궁금한 점은 Optional무엇입니까? 파이썬에 대해 어떤 변화가 있었습니까? 아니면 마음이 바뀌 었습니까? None기본값 으로 인해 여전히 엄격하게 필요하지 않습니까?
Praxeolitic

10
@Praxeolitic 예, 실제로 기본값으로 Optional사용할 때 자동, 암시 적 주석은 None특정 사용 사례를 어렵게 만들었으며 현재 PEP에서 제거되고 있습니다.
Martijn Pieters

5
관심있는 사람들을 위해 이것을 논의하는 링크가 있습니다. Optional미래에는 분명한 것이 필요할 것 같습니다.
Rick은 Monica를 지원합니다. Monica

이것은 실제로 Callable에서 지원되지 않습니다 : github.com/python/mypy/issues/5876
Shital Shah

1
@ ShitalShah : 그것은 실제로 그 문제에 관한 것이 아닙니다. Callable지원하지 않는 어떤 유형에 대한 힌트의 언급 *args이나 **kwargs 마침표를 . 이 특정 문제는 특정 인수 와 임의의 수의 다른 인수를 허용하는 호출 가능 항목을 표시하는 것과 관련이 있으므로 *args: Any, **kwargs: Any두 포괄 모두에게 매우 구체적인 유형 힌트를 사용하십시오. 더 구체적으로 설정 *args하거나 설정 한 경우에는를 **kwargs사용할 수 있습니다 Protocol.
Martijn Pieters

26

이를 수행하는 올바른 방법은 @overload

from typing import overload

@overload
def foo(arg1: int, arg2: int) -> int:
    ...

@overload
def foo(arg: int) -> int:
    ...

def foo(*args):
    try:
        i, j = args
        return i + j
    except ValueError:
        assert len(args) == 1
        i = args[0]
        return i

print(foo(1))
print(foo(1, 2))

@overload실제 구현에 주석을 추가 하거나 입력 하지 마십시오 .

스텁 파일 외부에서typing @overload를 지원하려면 새로운 버전 과 mypy가 모두 필요 합니다 .

또한이를 사용하여 어떤 인수 유형이 어떤 리턴 유형에 해당하는지 명시 적으로 표시하는 방식으로 리턴 된 결과를 변경할 수 있습니다. 예 :

from typing import Tuple, overload

@overload
def foo(arg1: int, arg2: int) -> Tuple[int, int]:
    ...

@overload
def foo(arg: int) -> int:
    ...

def foo(*args):
    try:
        i, j = args
        return j, i
    except ValueError:
        assert len(args) == 1
        i = args[0]
        return i

print(foo(1))
print(foo(1, 2))

2
나는이 대답이 더 일반적인 경우를 다루기 때문에 좋아합니다. 되돌아 보면, 예제로 (type1)vs (type1, type1)함수 호출을 사용해서는 안됩니다 . 어쩌면 (type1)vs (type2, type1)가 더 좋은 예 였을 것이고 내가 왜이 대답을 좋아하는지 보여줍니다. 이것은 또한 다른 리턴 유형을 허용합니다. 그러나, 특별한 경우에 당신은 단지 하나 개의 반환 유형이 당신을 어디에 *args*kwargs모두 같은 유형, Martjin의 대답의 기술은 모두 답변을 유용 그래서 더 의미가 있습니다.
Praxeolitic

4
그러나 *args최대 개수의 인수가있는 곳 (2)이 여전히 잘못되었습니다 .
Martijn Pieters

1
@MartijnPieters 왜 *args여기 가 왜 틀렸습니까 ? 예상 호출이 (type1)vs (type2, type1)인 경우 인수 수는 가변적이며 후행 인수에 대한 적절한 기본값이 없습니다. 최대 값이 있어야하는 이유는 무엇입니까?
Praxeolitic

1
*args실제로는 0 개 이상 , 캐치되지 않은 균질 한 인수, 또는 '손길이 닿지 않은 캐치 올 (catchall)'을 전달하기위한 것입니다. 하나의 필수 인수와 하나의 옵션이 있습니다. 그것은 완전히 다르며 일반적으로 두 번째 인수에 센티넬 기본값을 지정하여 처리됩니다.
Martijn Pieters

3
PEP를 살펴본 후에는 @overload를 의도적으로 사용하지 않습니다. 이 답변은의 유형에 개별적으로 주석을 달 수있는 흥미로운 방법을 보여 주지만 *args질문에 대한 더 나은 답변은 이것이 전혀 수행되어야하는 것이 아니라는 것입니다.
Praxeolitic

20

당신이 주석 대신 유형을 추가하는 데 사용 댓글 파이썬에서이 개 파일과 필요성을 mypy를 사용하려는 경우 이전의 대답에 짧은 추가로, 당신은에 대한 유형을 접두사 필요가 argskwargs***각각 :

def foo(param, *args, **kwargs):
    # type: (bool, *str, **int) -> None
    pass

이것은 mypy에 의해 아래의 파이썬 3.5 버전과 동일하게 취급됩니다 foo:

def foo(param: bool, *args: str, **kwargs: int) -> None:
    pass
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.