avatar
Published on

Array vs ArrayLike, Promise vs PromiseLike

Author
  • avatar
    Name
    yceffort

ํƒ€์ž…์Šคํฌ๋ฆฝํŠธ์—๋Š” ArrayLike๋ผ๋Š”๊ฒŒ ์กด์žฌํ•œ๋‹ค. Array๋Š” ์ผ๋ฐ˜์ ์ธ ๋ฐฐ์—ด์„ ์˜๋ฏธํ•˜๋Š”๋ฐ, ArrayLike๋Š” ๋ฌด์—‡์ผ๊นŒ? ์ด๋ฅผ ์•Œ์•„๋ณด๊ธฐ ์œ„ํ•ด lib.es5.d.ts์— ๊ฐ€์„œ ๊ฐ๊ฐ์˜ ์ŠคํŽ™์„ ์‚ดํŽด๋ณด์ž.

Array

ArrayLike<T>

interface ArrayLike<T> {
  readonly length: number
  readonly [n: number]: T
}

Array<T>

interface Array<T> {
  /**
   * Returns the value of the first element in the array where predicate is true, and undefined
   * otherwise.
   * @param predicate find calls predicate once for each element of the array, in ascending
   * order, until it finds one where predicate returns true. If such an element is found, find
   * immediately returns that element value. Otherwise, find returns undefined.
   * @param thisArg If provided, it will be used as the this value for each invocation of
   * predicate. If it is not provided, undefined is used instead.
   */
  find<S extends T>(
    predicate: (this: void, value: T, index: number, obj: T[]) => value is S,
    thisArg?: any,
  ): S | undefined
  find(
    predicate: (value: T, index: number, obj: T[]) => unknown,
    thisArg?: any,
  ): T | undefined

  /**
   * Returns the index of the first element in the array where predicate is true, and -1
   * otherwise.
   * @param predicate find calls predicate once for each element of the array, in ascending
   * order, until it finds one where predicate returns true. If such an element is found,
   * findIndex immediately returns that element index. Otherwise, findIndex returns -1.
   * @param thisArg If provided, it will be used as the this value for each invocation of
   * predicate. If it is not provided, undefined is used instead.
   */
  findIndex(
    predicate: (value: T, index: number, obj: T[]) => unknown,
    thisArg?: any,
  ): number

  /**
   * Changes all array elements from `start` to `end` index to a static `value` and returns the modified array
   * @param value value to fill array section with
   * @param start index to start filling the array at. If start is negative, it is treated as
   * length+start where length is the length of the array.
   * @param end index to stop filling the array at. If end is negative, it is treated as
   * length+end.
   */
  fill(value: T, start?: number, end?: number): this

  /**
   * Returns the this object after copying a section of the array identified by start and end
   * to the same array starting at position target
   * @param target If target is negative, it is treated as length+target where length is the
   * length of the array.
   * @param start If start is negative, it is treated as length+start. If end is negative, it
   * is treated as length+end.
   * @param end If not specified, length of the this object is used as its default value.
   */
  copyWithin(target: number, start: number, end?: number): this
}

Array๋Š” ๋”ฑ๋ด๋„ ์šฐ๋ฆฌ๊ฐ€ ์ผ๋ฐ˜์ ์œผ๋กœ ์•„๋Š” ๋ฐฐ์—ด์— ๋“ค์–ด๊ฐ€๋Š” ๋ฉ”์†Œ๋“œ๋“ค์ด ์ •์˜๋˜์–ด ์žˆ์ง€๋งŒ, ArrayLike๋Š” ๊ทธ๋ ‡์ง€ ์•Š๋‹ค. length์™€ index๋กœ๋งŒ ์ ‘๊ทผํ•  ์ˆ˜ ์žˆ๋„๋ก ๊ตฌํ˜„๋˜์–ด ์žˆ๋‹ค. ์ด๋Š” ๋ฐ”๋กœ ์šฐ๋ฆฌ๊ฐ€ ์ž˜ ์•Œ๊ณ  ์žˆ๋Š” ์œ ์‚ฌ ๋ฐฐ์—ด ๊ฐ์ฒด๋‹ค. ๋ฐฐ์—ด ์ฒ˜๋Ÿผ ์ˆœํšŒํ•  ์ˆ˜ ์žˆ์ง€๋งŒ, ๊ทธ ๋ฟ์ธ ์œ ์‚ฌ ๋ฐฐ์—ด ๊ฐ์ฒด. ๋Œ€ํ‘œ์ ์œผ๋กœ๋Š”

๊ฐ€ ์žˆ๋‹ค.

Promise

๊ทธ๋ ‡๋‹ค๋ฉด ์ด๋ฒˆ์—๋Š” Promise๋ฅผ ์‚ดํŽด๋ณด์ž.

Promise<T> (lib.2018.promise.d.ts)

interface Promise<T> {
  /**
   * Attaches a callback that is invoked when the Promise is settled (fulfilled or rejected). The
   * resolved value cannot be modified from the callback.
   * @param onfinally The callback to execute when the Promise is settled (fulfilled or rejected).
   * @returns A Promise for the completion of the callback.
   */
  finally(onfinally?: (() => void) | undefined | null): Promise<T>
}

Promise<T> (lib.es5.d.ts)

interface Promise<T> {
  /**
   * Attaches callbacks for the resolution and/or rejection of the Promise.
   * @param onfulfilled The callback to execute when the Promise is resolved.
   * @param onrejected The callback to execute when the Promise is rejected.
   * @returns A Promise for the completion of which ever callback is executed.
   */
  then<TResult1 = T, TResult2 = never>(
    onfulfilled?:
      | ((value: T) => TResult1 | PromiseLike<TResult1>)
      | undefined
      | null,
    onrejected?:
      | ((reason: any) => TResult2 | PromiseLike<TResult2>)
      | undefined
      | null,
  ): Promise<TResult1 | TResult2>

  /**
   * Attaches a callback for only the rejection of the Promise.
   * @param onrejected The callback to execute when the Promise is rejected.
   * @returns A Promise for the completion of the callback.
   */
  catch<TResult = never>(
    onrejected?:
      | ((reason: any) => TResult | PromiseLike<TResult>)
      | undefined
      | null,
  ): Promise<T | TResult>
}

PromiseLike<T>

interface PromiseLike<T> {
  /**
   * Attaches callbacks for the resolution and/or rejection of the Promise.
   * @param onfulfilled The callback to execute when the Promise is resolved.
   * @param onrejected The callback to execute when the Promise is rejected.
   * @returns A Promise for the completion of which ever callback is executed.
   */
  then<TResult1 = T, TResult2 = never>(
    onfulfilled?:
      | ((value: T) => TResult1 | PromiseLike<TResult1>)
      | undefined
      | null,
    onrejected?:
      | ((reason: any) => TResult2 | PromiseLike<TResult2>)
      | undefined
      | null,
  ): PromiseLike<TResult1 | TResult2>
}

Promise<T>์—๋Š” finally๋งŒ ์žˆ๊ณ , PromiseLike<T>์—๋Š” then ๋ฐ–์— ์—†๋‹ค. ๐Ÿค” ์ด ๋‘˜์˜ ์ฐจ์ด๋ฅผ ๋จผ์ € ์•Œ ํ•„์š”๊ฐ€ ์žˆ๋‹ค.

then vs finally

  • finally: promise๊ฐ€ ์ฒ˜๋ฆฌ๋˜๋ฉด ์ถฉ์กฑ๋˜๊ฑฐ๋‚˜ (resolve) ๊ฑฐ๋ถ€๋˜๊ฑฐ๋‚˜ (reject) ์ƒ๊ด€์—†์ด ์‹คํ–‰ํ•˜๋Š” ์ฝœ๋ฐฑํ•จ์ˆ˜๋‹ค. Promise์˜ ์„ฑ๊ณต์ ์œผ๋กœ ์ˆ˜ํ–‰๋˜์—ˆ๋Š”์ง€, ๊ฑฐ์ ˆ๋˜์—ˆ๋Š”์ง€์— ๊ด€๊ณ„์—†์ด Promise๊ฐ€ ์ฒ˜๋ฆฌ๋œ ํ›„์— ๋ฌด์กฐ๊ฑด ํ•œ๋ฒˆ์€ ์‹คํ–‰๋˜๋Š” ์ฝ”๋“œ๋‹ค.
  • then: ์€ ์šฐ๋ฆฌ๊ฐ€ ์ž˜ ์•„๋Š” ๊ฒƒ์ฒ˜๋Ÿผ Promise๋ฅผ ๋ฆฌํ„ดํ•˜๊ณ  ๋‘๊ฐœ์˜ ์ฝœ๋ฐฑํ•จ์ˆ˜๋ฅผ ๋ฐ›๋Š”๋‹ค. ํ•˜๋‚˜๋Š” ์ถฉ์กฑ๋˜์—ˆ์„ ๋•Œ (resolve) ๊ทธ๋ฆฌ๊ณ  ๊ฑฐ๋ถ€๋˜์—ˆ์„ ๋•Œ (reject)๋ฅผ ์œ„ํ•œ ์ฝœ๋ฐฑ ํ•จ์ˆ˜๋‹ค.
p.then(onFulfilled, onRejected)

p.then(
  function (value) {
    // ์ดํ–‰
  },
  function (reason) {
    // ๊ฑฐ๋ถ€
  },
)

๊ทธ๋ฆฌ๊ณ  ๋˜ํ•œ๊ฐ€์ง€๋Š” finally๋Š” Promise ์ฒด์ด๋‹์—์„œ ๊ฒฐ๊ณผ๋ฅผ ๋ฐ›์„ ์ˆ˜ ์—†๋‹ค๋Š” ๊ฒƒ์ด๋‹ค.

const result = new Promise((resolve, reject) => resolve(10))
  .then((x) => {
    console.log(x) // 10
    return x + 1
  })
  .finally((x) => {
    console.log(x) // undefined
    return x + 2
  })
// then์—์„œ ๋ฆฌํ„ดํ–ˆ๋˜ 11์„ resolve ํ•œ๋‹ค.
result // Promiseย {<fulfilled>: 11}

๋˜๋‹ค๋ฅธ ์ฐจ์ด๋Š” ์—๋Ÿฌํ•ธ๋“ค๋ง๊ณผ Promise chaining์ด๋‹ค. ๋งŒ์•ฝ promise chaining์—์„œ ์—๋Ÿฌ์ฒ˜๋ฆฌ๋ฅผ ๋ฏธ๋ฃจ๊ณ  ๋‹ค๋ฅธ ์–ด๋”˜๊ฐ€์—์„œ ์ฒ˜๋ฆฌํ•˜๊ณ  ์‹ถ๋‹ค๋ฉด, finally๋ฅผ ์‚ฌ์šฉํ•˜๋ฉด ๋œ๋‹ค.

new Promise((resolve, reject) => reject(0))
  .catch((x) => {
    console.log(x) // 0
    throw x
  })
  .then((x) => {
    console.log(x) // Will not run
  })
  .finally(() => {
    console.log('clean up') // 'clean up'
  })
// Uncaught (in promise) 0
// try catch ๋กœ ์žก์œผ๋ฉด ์žกํžŒ๋‹ค!

๋์œผ๋กœ finally๋Š” es2018์—์„œ ๋‚˜์˜จ ๋ฉ”์†Œ๋“œ ์ด๊ธฐ ๋•Œ๋ฌธ์— lib.es2018.promise.d.ts์— ์กด์žฌํ•œ๋‹ค. https://2ality.com/2017/07/promise-prototype-finally.html

์•„๋ฌดํŠผ ๋‹ค์‹œ ๋Œ์•„๊ฐ€์„œ, catch๊ฐ€ ์—†๋Š” PromiseLike๋Š” ์™œ ์กด์žฌํ•˜๋Š” ๊ฒƒ์ผ๊นŒ? ๐Ÿค” Promise๊ฐ€ ์ •์‹ ์ŠคํŽ™์ด ๋˜๊ธฐ ์ „, Promise๋ฅผ ๊ตฌํ˜„ํ•˜๊ธฐ ์œ„ํ•œ ๋‹ค์–‘ํ•œ ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ๊ฐ€ ์กด์žฌํ–ˆ๋‹ค.

์ด๋“ค์€ ํ‘œ์ค€์ด์ „์— ํƒœ์–ด๋‚˜ catch ๊ตฌ๋ฌธ์—†์ด promise๋ฅผ ์ฒ˜๋ฆฌํ•˜๊ณ  ์žˆ์—ˆ๊ณ , ํƒ€์ž…์Šคํฌ๋ฆฝํŠธ๋Š” ์ด๋ฅผ ์ง€์›ํ•˜๊ธฐ ์œ„ํ•ด์„œ PromiseLike๋ฅผ ๋งŒ๋“  ๊ฒƒ์ด์—ˆ๋‹ค.

๋”ฐ๋ผ์„œ Promise ๋ฟ๋งŒ ์•„๋‹ˆ๋ผ ์ข€๋” ๊ด‘์˜์˜ Promise (ํ‘œ์ค€ ์ด์ „์— ๋งŒ๋“ค์–ด์ง„ ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ๋กœ ๋งŒ๋“ค์–ด์ง„ Promise)๋ฅผ ์ฒ˜๋ฆฌํ•˜๊ธฐ ์œ„ํ•ด์„œ PromiseLike ํƒ€์ž…์„ ์ถ”๊ฐ€ํ•˜๊ฒŒ ๋œ ๊ฒƒ์ด๋‹ค.