일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
- c# 윈폼
- C# delegate
- vim
- Flutter
- Data Structure
- Python
- C언어 포인터
- c# winform
- 포인터
- 다트 언어
- 깃
- git
- 유니티
- dart 언어
- gitlab
- c언어
- HTML
- C++
- jupyter lab
- Algorithm
- github
- Houdini
- 구조체
- 플러터
- c# 추상 클래스
- docker
- c#
- 도커
- jupyter
- Unity
- Today
- Total
nomad-programmer
[Programming/Python] 이터레이터 (iterator) 본문
__iter__, __next__ 메서드를 구현하여 직접 이터레이터를 만들어보자. 간단하게 range(횟수)처럼 동작하는 이터레이터다.
class Counter:
def __init__(self, stop: int) -> None:
self.__current = 0
self.__stop = stop
def __iter__(self):
return self
def __next__(self) -> int:
if self.__current < self.__stop:
r = self.__current
self.__current += 1
return r
else:
raise StopIteration
if __name__ == '__main__':
for i in Counter(3):
print(i, end=' ')
# 결과
0 1 2
__iter__ 메서드를 만드는데 여기서는 self만 반환한다. 이 객체는 리스트, 문자열, 딕셔너리, 세트, range처럼 __iter__ 를 호출해줄 반복 가능한 객체(iterable)가 없으므로 현재 인스턴스를 반환하면 된다. 즉, 이 객체는 반복 가능한 객체이면서 이터레이터이다.
그 다음에 __next__ 메서드를 만든다. __next__ 에서는 조건에 따라 숫자를 만들어내거나 StopIteration 예외를 발생시킨다.
인덱스로 접근할 수 있는 이터레이터
class Counter:
def __init__(self, stop: int) -> None:
self.__current = 0
self.__stop = stop
def __iter__(self):
return self
def __next__(self) -> int:
if self.__current < self.__stop:
r = self.__current
self.__current += 1
return r
raise StopIteration
def __getitem__(self, item: int) -> int:
if item < self.__stop:
return item
raise IndexError
if __name__ == '__main__':
print(Counter(3)[2])
# 결과
2
__getitem__ 메서드를 구현하면 인덱스로 접근할 수 있는 이터레이터를 만들 수 있다.
iter, next 함수 활용
파이썬 내장 함수 iter, next에 대해 알아보자. iter는 객체의 __iter__ 메서드를 호출해주고, next는 객체의 __next__ 메서드를 호출해준다.
>>> it = iter(range(3))
>>> next(it)
0
>>> next(it)
1
...
반복 가능한 객체에서 __iter__를 호출하고 이터레이터에서 __next__ 메서드를 호출한 것과 똑같다. 즉, iter는 반복 가능한 객체에서 이터레이터를 반환하고 next는 이터레이터에서 값을 차례대로 꺼낸다. iter와 next는 이런 기능 이외에도 다양한 방식으로 사용할 수 있다.
iter
iter는 반복을 끝낼 값을 지정하면 특정 값이 나올 때 반복을 끝낸다. 이 경우에는 반복 가능한 객체 대신 호출 가능한 객체(callable)를 넣어준다. 참고로 반복을 끝낸 값은 sentinel이라고 부르는데 감시병이라는 뜻이다. 즉, 반복을 감시하다가 특정 값이 나오면 반복을 끝낸다고 해서 sentinel이다.
- iter(호출가능한객체, 반복을끝낼값)
예를 들어 random.randint(0, 5)와 같이 0부터 5까지 무작위로 숫자를 생성할 때 2가 나오면 반복을 끝내도록 만들 수 있다. 이때 호출 가능한 객체를 넣어야 하므로 매개변수가 없는 함수 또는 람다 표현식으로 만들어준다.
>>> import random
>>> it = iter(lambda: random.randint(0, 5), 2)
>>> next(it)
0
>>> next(it)
3
>>> next(it)
# 만약 여기서 2가 나왔다면 StopIteration 예외가 발생하여 반복 종료
next(it)로 숫자를 계속 만들다가 2가 나오면 StopIteration이 발생한다. 다음과 같이 for 반복문에 넣어서 사용할 수도 있다.
mport random
if __name__ == '__main__':
for i in iter(lambda: random.randint(0, 5), 2):
print(i, end=' ')
# 결과
1 1 0 5 3 3 5 4
이렇게 iter 함수를 활용하면 if 조건문으로 매번 숫자가 2인지 검사하지 않아도 되므로 코드가 좀 더 간단해진다.
next
next는 기본 값을 지정할 수 있다. 기본 값을 지정하면 반복이 끝나더라도 StopIteration이 발생하지 않고 기본값을 출력한다. 즉, 반복할 수 있을 때는 해당 값을 출력하고, 반복이 끝났을 때는 기본값을 출력한다. 다음은 range(3)으로 0, 1, 2 세 번 반복하는데 next에 기본값으로 10을 지정했다.
- next(반복가능한객체, 기본값)
>>> it = iter(range(3))
>>> next(it, 10)
0
>>> next(it, 10)
1
>>> next(it, 10)
2
>>> next(it, 10)
10
이터레이터를 만들 때 __iter__, __next__ 또는 __getitem__ 메서드를 구현해야 한다는 점을 기억해야 한다.
'Programming > Python' 카테고리의 다른 글
[Programming/Python] 코루틴 (coroutine) (2) | 2023.01.29 |
---|---|
[Programming/Python] 제네레이터 (generator) (0) | 2023.01.29 |
[Programming/Python] yield from으로 값을 여러 번 바깥으로 전달하기 (0) | 2021.09.20 |
[Programming/Python] 단위 테스트(unittest) - 보다 견고한 코드 만들기 (0) | 2021.05.26 |
[Programming/Python] Class Method & Static Method (0) | 2021.02.08 |