Notice
Recent Posts
Recent Comments
Link
«   2025/01   »
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
Archives
Today
Total
관리 메뉴

nomad-programmer

[Programming/Python] 이터레이터 (iterator) 본문

Programming/Python

[Programming/Python] 이터레이터 (iterator)

scii 2023. 1. 29. 19:47

__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__ 메서드를 구현해야 한다는 점을 기억해야 한다.

Comments