TypeScript/ํƒ€์ž…์Šคํฌ๋ฆฝํŠธ TypeScript

[ํƒ€์ž…์Šคํฌ๋ฆฝํŠธ] ์ œ๋„ˆ๋ ˆ์ดํ„ฐ, ๋ฐ˜๋ณต์ž(iterator) ์ถ”๋ก ๊ณผ ๋ช…์‹œ

Rainbow๐ŸŒˆCoder 2022. 4. 20. 12:11
728x90

Generator

ํƒ€์ž…์Šคํฌ๋ฆฝํŠธ๋Š” Generator๊ฐ€ yield๋ฅผ ํ†ตํ•ด ๋ฐฉ์ถœํ•œ ๊ฐ’์„ ํ† ๋Œ€๋กœ ๋ฐ˜๋ณต์ž์˜ ํƒ€์ž…์„ ์ถ”๋ก ํ•œ๋‹ค.

์ œ๋„ˆ๋ ˆ์ดํ„ฐ ํ•จ์ˆ˜

๋ฐ˜๋ณต์ž(iterator)์™€ ์ œ๋„ˆ๋ ˆ์ดํ„ฐ๋Š” ์ƒ์ƒ๊ด€๊ณ„๋‹ค.

์ œ๋„ˆ๋ ˆ์ดํ„ฐ๋กœ ๊ฐ’์˜ ์ŠคํŠธ๋ฆผ์„ ์ƒ์„ฑํ•  ์ˆ˜ ์žˆ๊ณ  ๋ฐ˜๋ณต์ž๋กœ ์ƒ์„ฑ๋œ ๊ฐ’์„ ์†Œ๋น„ํ•  ์ˆ˜ ์žˆ๊ธฐ ๋•Œ๋ฌธ์ด๋‹ค.

์ดํ„ฐ๋Ÿฌ๋ธ”(iterable;๋ฐ˜๋ณตํ•  ์ˆ˜ ์žˆ๋Š”)Symbol.iterator ๋ผ๋Š” ํ”„๋กœํผํ‹ฐ(๋ฐ˜๋ณต์ž๋ฅผ ๋ฐ˜ํ™˜ํ•˜๋Š” ํ•จ์ˆ˜)๋ฅผ ๊ฐ€์ง„ ๋ชจ๋“  ๊ฐ์ฒด

๋ฐ˜๋ณต์ž(iterator)next๋ผ๋Š” ๋ฉ”์„œ๋“œ(value, done ๋‘ ํ”„๋กœํผํ‹ฐ๋ฅผ ๊ฐ€์ง„ ๊ฐ์ฒด๋ฅผ ๋ฐ˜ํ™˜)๋ฅผ ์ •์˜ํ•œ ๊ฐ์ฒด

 

์ž๋ฐ”์Šคํฌ๋ฆฝํŠธ์—์„œ ์ œ๋„ˆ๋ ˆ์ดํ„ฐ ํ•จ์ˆ˜๋ฅผ ํ˜ธ์ถœํ•˜๋ฉด ์ดํ„ฐ๋Ÿฌ๋ธ” ๋ฐ˜๋ณต์ž(IterableIterator)๊ฐ€ ๋ฐ˜ํ™˜๋œ๋‹ค.

ํƒ€์ž…์Šคํฌ๋ฆฝํŠธ์—์„œ๋„ ์ง€์›ํ•œ๋‹ค.

//๋‹ค์Œ ํ”ผ๋ณด๋‚˜์น˜ ์ˆซ์ž๋ฅผ ๊ณ„์‚ฐํ•˜๊ธฐ ์œ„ํ•ด a์— b๋ฅผ, b์— a+b๋ฅผ ํ•œ๋ฒˆ์— ๋‹ค์‹œ ํ• ๋‹นํ•œ๋‹ค.

function* createFibonacciGenerator() {
  let a = 0;
  let b = 1;
  while (true) {
    yield a;
    [a, b] = [b, a + b];
  }
}

let fiboancciGenerator = createFibonacciGenerator();
                                        //IterableIterator<number>
console.log(fiboancciGenerator.next()); //{ value: 0, done: false }
console.log(fiboancciGenerator.next()); //{ value: 1, done: false }
console.log(fiboancciGenerator.next()); //{ value: 1, done: false }
console.log(fiboancciGenerator.next()); //{ value: 2, done: false }
console.log(fiboancciGenerator.next()); //{ value: 3, done: false }
console.log(fiboancciGenerator.next()); //{ value: 5, done: false }
console.log(fiboancciGenerator.next()); //{ value: 8, done: false }
console.log(fiboancciGenerator.next()); //{ value: 13, done: false }

createFibonacciGenerator ๋ฅผ ํ˜ธ์ถœํ•˜๋ฉด Symbol.iterator ํ”„๋กœํผํ‹ฐ์™€ next ๋ฉ”์„œ๋“œ๋ฅผ ๋ชจ๋‘ ์ •์˜ํ•œ ๊ฐ’์„ ์–ป๊ฒŒ ๋œ๋‹ค. ์ฆ‰, ์ดํ„ฐ๋Ÿฌ๋ธ”๊ณผ ๋ฐ˜๋ณต์ž ๋‘ ๊ฐ€์ง€๊ฐ€ ๊ฒฐํ•ฉ๋œ ์ œ๋„ˆ๋ ˆ์ดํ„ฐ๊ฐ€ ๋ฐ˜ํ™˜๋œ๋‹ค.

์œ„ ์ฝ”๋“œ์˜ createFibonacciGenerator ํ•จ์ˆ˜๋Š” IterableIterator์„ ๋ฐ˜ํ™˜ํ•˜๊ณ , ์ด ์ œ๋„ˆ๋ ˆ์ดํ„ฐ์— next๋ฅผ ํ˜ธ์ถœํ•  ๋•Œ๋งˆ๋‹ค ๋‹ค์Œ ํ”ผ๋ณด๋‚˜์น˜ ๊ฐ’์„ ๊ณ„์‚ฐํ•ด์„œ ๊ฒฐ๊ณผ๋ฅผ ๋ฐฉ์ถœํ•œ๋‹ค. ํƒ€์ž…์Šคํฌ๋ฆฝํŠธ๋Š” Generator๊ฐ€ yield๋ฅผ ํ†ตํ•ด ๋ฐฉ์ถœํ•œ ๊ฐ’์„ ํ† ๋Œ€๋กœ ๋ฐ˜๋ณต์ž์˜ ํƒ€์ž…์„ ์ถ”๋ก ํ•œ๋‹ค.

fiboancciGenerator๋ฅผ ํ˜ธ์ถœํ•˜๋ฉด Symbol.iterator ํ”„๋กœํผํ‹ฐ์™€ next ๋ฉ”์„œ๋“œ๋ฅผ ๋ชจ๋‘ ์ •์˜ํ•œ ๊ฐ’์„ ์–ป๊ฒŒ ๋œ๋‹ค. ์ฆ‰, ์ดํ„ฐ๋Ÿฌ๋ธ”๊ณผ ๋ฐ˜๋ณต์ž ๋‘๊ฐ€์ง€๊ฐ€ ๊ฒฐํ•ฉ๋œ ์ œ๋„ˆ๋ ˆ์ดํ„ฐ๊ฐ€ ๋ฐ˜ํ™˜๋œ๋‹ค.

๋‹ค์Œ ์˜ˆ์ฒ˜๋Ÿผ IterableIterator ์—์„œ ๋ฐฉ์ถœํ•˜๋Š” ํƒ€์ž…์„ ๊ฐ์‹ธ์„œ ์ œ๋„ˆ๋ ˆ์ดํ„ฐ ๋ฐ˜ํ™˜ ํƒ€์ž…์„ ๋ช…์‹œํ•˜๋Š” ๋ฐฉ๋ฒ•๋„ ์žˆ๋‹ค.

์ง์ ‘ ๋ช…์‹œํ•  ์ˆ˜๋„ ์žˆ๋‹ค.

 
function* createNumbers(): IterableIterator<number> {
    let n = 0;
    while( 1 ){
        yield n++;
    }
}
let numbers = creatNumbers();
numbers.next() // {value: 0, done: false}
numbers.next() // {value: 1, done: false}
numbers.next() // {value: 2, done: false}

 

๋ฐ˜๋ณต์ž(iterator)์™€ ์ œ๋„ˆ๋ ˆ์ดํ„ฐ(Generator๋Š” ํ˜ธ์ถœ๋  ๋•Œ Iterator๋ฅผ ์ž๋™์œผ๋กœ ์ƒ์„ฑํ•˜๋Š” ํŠน์ˆ˜ ํ•จ์ˆ˜)๋Š” ์ƒ์ƒ๊ด€๊ณ„๋‹ค.

์ œ๋„ˆ๋ ˆ์ดํ„ฐ๋กœ ๊ฐ’์˜ ์ŠคํŠธ๋ฆผ์„ ์ƒ์„ฑํ•  ์ˆ˜ ์žˆ๊ณ  ๋ฐ˜๋ณต์ž๋กœ ์ƒ์„ฑ๋œ ๊ฐ’์„ ์†Œ๋น„ํ•  ์ˆ˜ ์žˆ๊ธฐ ๋•Œ๋ฌธ์ด๋‹ค.

 

 

์ดํ„ฐ๋Ÿฌ๋ธ”(iterable;๋ฐ˜๋ณตํ•  ์ˆ˜ ์žˆ๋Š”)

Symbol.iterator ๋ผ๋Š” ํ”„๋กœํผํ‹ฐ(๋ฐ˜๋ณต์ž๋ฅผ ๋ฐ˜ํ™˜ํ•˜๋Š” ํ•จ์ˆ˜)๋ฅผ ๊ฐ€์ง„ ๋ชจ๋“  ๊ฐ์ฒด

- Symbol.Iterator ๋ฉ”์„œ๋“œ๊ฐ€ ๊ตฌํ˜„๋˜์–ด ์žˆ์–ด์•ผ ํ•œ๋‹ค. ์ฆ‰, ๊ฐ์ฒด๊ฐ€ Symbol.iterator ํ‚ค๋ฅผ ๊ฐ–๋Š” ์†์„ฑ์ด ์žˆ์–ด์•ผ ํ•จ
- Symbol.Iterator ๋Š” Iterator์„ ๋ฐ˜ํ™˜ํ•ด์•ผ ํ•œ๋‹ค

 

๋ฐ˜๋ณต์ž(iterator)

next๋ผ๋Š” ๋ฉ”์„œ๋“œ(value, done ๋‘ ํ”„๋กœํผํ‹ฐ๋ฅผ ๊ฐ€์ง„ ๊ฐ์ฒด๋ฅผ ๋ฐ˜ํ™˜)๋ฅผ ์ •์˜ํ•œ ๊ฐ์ฒด

- next ๋ฉ”์„œ๋“œ๋ฅผ ๊ฐ€์ง„๋‹ค
- next ๋ฉ”์„œ๋“œ๋Š” value์™€ done ์†์„ฑ์„ ๊ฐ€์ง„ ๊ฐ์ฒด๋ฅผ ๋ฐ˜ํ™˜ํ•œ๋‹ค.
- ์ž‘์—…์ด ๋๋‚˜๋ฉด done์€ true๊ฐ€ ๋œ๋‹ค.

 

 

๊ฐ€๋ น createFibonacciGenerator ํ•จ์ˆ˜๋ฅผ ํ˜ธ์ถœํ•˜๋ฉด Symbol.iterator ํ”„๋กœํผํ‹ฐ์™€ next ๋ฉ”์„œ๋“œ๋ฅผ ๋ชจ๋‘ ์ •์˜ํ•œ ๊ฐ’์„ ์–ป๊ฒŒ ๋œ๋‹ค. ์ฆ‰, ์ดํ„ฐ๋Ÿฌ๋ธ”๊ณผ ๋ฐ˜๋ณต์ž ๋‘ ๊ฐ€์ง€๊ฐ€ ๊ฒฐํ•ฉ๋œ ์ œ๋„ˆ๋ ˆ์ดํ„ฐ๊ฐ€ ๋ฐ˜ํ™˜๋œ๋‹ค.

Symbol.iterator์™€ next๋ฅผ ๊ตฌํ˜„ํ•˜๋Š” ๊ฐ์ฒด(๋˜๋Š” ํด๋ž˜์Šค)๋ฅผ ๋งŒ๋“ค์–ด ๋ฐ˜๋ณต์ž๋‚˜ ์ดํ„ฐ๋Ÿฌ๋ธ”์„ ์ง์ ‘ ์ •์˜ํ•  ์ˆ˜ ์žˆ๋‹ค. ๋‹ค์Œ์€ 1์—์„œ 10๊นŒ์ง€์˜ ์ˆซ์ž๋ฅผ ๋ฐ˜๋ณตํ•˜๋Š” ๋ฐ˜๋ณต์ž๋ฅผ ์ •์˜ํ•˜๋Š” ์˜ˆ๋‹ค.

  let numbers = {
    *[Symbol.iterator]() {
      for (let n = 1; n <= 10; n++) {
        yield n;
      }
    },
  };

์‚ฌ์šฉ์ž ์ •์˜ iterable

Symbol.iterator์™€ next๋ฅผ ๊ตฌํ˜„ํ•˜๋Š” ๊ฐ์ฒด(๋˜๋Š” ํด๋ž˜์Šค)๋ฅผ ๋งŒ๋“ค์–ด ๋ฐ˜๋ณต์ž๋‚˜ ์ดํ„ฐ๋Ÿฌ๋ธ”์„ ์ง์ ‘ ์ •์˜ํ•  ์ˆ˜ ์žˆ๋‹ค.

๋‹ค์Œ์€ 1์—์„œ 10๊นŒ์ง€์˜ ์ˆซ์ž๋ฅผ ๋ฐ˜๋ณตํ•˜๋Š” ๋ฐ˜๋ณต์ž๋ฅผ ์ •์˜ํ•˜๋Š” ์˜ˆ

์ด์™€ ๊ฐ™์ด ์ž์‹ ์˜ ๋ฐ˜๋ณต๊ฐ€๋Šฅ ๊ฐ์ฒด๋ฅผ ๋งŒ๋“ค ์ˆ˜ ์žˆ๋‹ค

let numbers = {
  *[Symbol.iterator]() {
    for (let n = 1; n <= 10; n++) {
      yield n;
    }
  },
};

console.log([...numbers]);
<์ถœ๋ ฅ>
[
  1, 2, 3, 4,  5,
  6, 7, 8, 9, 10
]

 

//์ปค์Šคํ…€ ๋ฐ˜๋ณต์ž
let numbers = {
  *[Symbol.iterator](){
    for(let n =1;n <= 10; n++){
      yield n;
    }
  }
}
์ด๋Ÿฌํ•œ ์‚ฌ์šฉ์ž ์ •์˜ ๋ฐ˜๋ณต๊ฐ€๋Šฅ ๊ฐ์ฒด์— ๋Œ€ํ•ด์„œ๋„ ํƒ€์ž…์Šคํฌ๋ฆฝํŠธ๊ฐ€ ์ œ๋„ˆ๋ ˆ์ดํ„ฐ ๋ฐ˜ํ™˜ ํƒ€์ž…์„ ์ถ”๋ก 

 

 

์ง์ ‘ ๋ช…์‹œ๋„ ๊ฐ€๋Šฅ

let numbers = {
  *[Symbol.iterator](): IterableIterator<number> {
    for (let n = 1; n <= 10; n++) {
      yield n;
    }
  },
};

for (let value of numbers) {
  console.log(value);
}

<์ถœ๋ ฅ>
1
2
3
4
5
6
7
8
9
10
 

 

<์ตœ์ข… ์ ๊ฒ€>

let numbers = {
  *[Symbol.iterator](): IterableIterator<number> {
    for (let n = 1; n <= 10; n++) {
      yield n;
    }
  },
};

//for-of๋กœ ๋ฐ˜๋ณต์ž ๋ฐ˜๋ณตํ•˜๊ธฐ
for (let a of numbers) {
  console.log(a);
}
<์ถœ๋ ฅ>
1
2
3
4
5
6
7
8
9
10

//๋ฐ˜๋ณต์ž ์Šคํ”„๋ ˆ๋“œ
let allNumbers = [...numbers];
console.log(allNumbers);

<์ถœ๋ ฅ>
[
  1, 2, 3, 4,  5,
  6, 7, 8, 9, 10
]

//๋ฐ˜๋ณต์ž ๊ตฌ์กฐ ๋ถ„ํ•ด ํ• ๋‹น(destructure)
let [one, two, ...rest] = numbers;
console.log(one);
console.log(two);
console.log(rest);

<์ถœ๋ ฅ>
1
2
[
  3, 4, 5,  6,
  7, 8, 9, 10
]

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

728x90