๐ŸŽพ ๊ธฐ์ˆ ์ฑ… ์Šคํ„ฐ๋””

23๋…„ 1์›”๋ถ€ํ„ฐ ํ™œ๋™ ์ค‘์ธ ๊ต์œก์—์„œ, ๋œป์ด ๋งž๋Š” ๋™๋ฃŒ๋“ค๊ณผ ํ•จ๊ป˜ ์ง„ํ–‰ํ•˜๊ฒŒ ๋œ ์Šคํ„ฐ๋””
์•ž์œผ๋กœ๋„ ๊พธ์ค€ํžˆ ๊ธฐ์ˆ  ์„œ์ ์„ ์ฝ๊ณ  ํ•จ๊ป˜ ๋ฐœ์ „ํ•˜๋Š” ์‹œ๊ฐ„์ด ๋˜์—ˆ์œผ๋ฉด ์ข‹๊ฒ ๋‹ค!

๋“ค์–ด๊ฐ€๊ธฐ์— ์•ž์„œ

์ด ๋‚ด์šฉ์€ ๊ฐœ๋ฐœ์ž ํ™ฉ์ค€์ผ - Vanilla Javascript๋กœ ์ƒํƒœ๊ด€๋ฆฌ ์‹œ์Šคํ…œ ๋งŒ๋“ค๊ธฐ์„ ๊ณต๋ถ€ํ•˜๋ฉฐ ์ž‘์„ฑํ•œ ๊ธ€์ž…๋‹ˆ๋‹ค. ๋Œ€๋ถ€๋ถ„์˜ ๋‚ด์šฉ์„ ํ™ฉ์ค€์ผ๋‹˜์˜ ๋ธ”๋กœ๊ทธ๋ฅผ ์ฐธ๊ณ ํ•˜์˜€๊ณ  ๋ช‡๊ฐœ์˜ ๊ฐœ๋… ๋‚ด์šฉ ์ •๋„๋งŒ ์ถ”๊ฐ€ ํ˜น์€ ๋‚ด์šฉ ์š”์•ฝ์ด ๋˜์–ด์žˆ์Šต๋‹ˆ๋‹ค. ์ž์„ธํ•œ ์‚ฌํ•ญ์€ ํ™ฉ์ค€์ผ๋‹˜์˜ ๋ธ”๋กœ๊ทธ๋ฅผ ์ฐธ๊ณ ํ•ด์ฃผ์„ธ์š”!(์ •๋ง ๋„ˆ๋ฌด ์ข‹์€ ๊ธ€์ด์—์š”~!)

Vanilla Javascript๋กœ ์ƒํƒœ๊ด€๋ฆฌ ์‹œ์Šคํ…œ ๋งŒ๋“ค๊ธฐ

1. ์ค‘์•™ ์ง‘์ค‘์‹ ์ƒํƒœ๊ด€๋ฆฌ

  • ํ˜„๋Œ€ ํ”„๋ก ํŠธ์—”๋“œ ๊ฐœ๋ฐœ์—์„œ ์ œ์ผ ์ค‘์š”ํ•œ ๊ฒƒ์€ ์ƒํƒœ๊ด€๋ฆฌ
  • ์ƒํƒœ๋ฅผ ๊ธฐ๋ฐ˜์œผ๋กœ DOM์„ ๋ Œ๋”๋งํ•˜๊ธฐ ๋•Œ๋ฌธ์ด๋‹ค.
  • ์–ดํ”Œ๋ฆฌ์ผ€์ด์…˜์˜ ๊ทœ๋ชจ๊ฐ€ ์ปค์งˆ์ˆ˜๋ก ์ปดํฌ๋„ŒํŠธ์˜ depth๊ฐ€ ๊นŠ์–ด์ง€๋ฉฐ ๋”๋ถˆ์–ด ์ƒํƒœ๊ด€๋ฆฌ๋„ ๊ต‰์žฅํžˆ ์–ด๋ ค์›Œ์ง„๋‹ค.

์ด ๋•Œ, ์ƒํƒœ๋ฅผ ์œ„์—์„œ ์•„๋ž˜๋กœ ํ•˜๋‚˜ํ•˜๋‚˜ ์ „๋‹ฌํ•˜์ง€ ์•Š๊ณ  ์ค‘์•™ ์ง‘์ค‘์†Œ ์—ญํ• ์„ ํ•˜๋ฉด์„œ ๋™์‹œ์— ์˜ˆ์ธก ๊ฐ€๋Šฅํ•œ ๋ฐฉ์‹์œผ๋กœ ๋‹ค๋ฃฐ ์ˆ˜ ์žˆ๋‹ค๋ฉด ์–ด๋–จ๊นŒ?

2. Observer Pattern์— ๋Œ€ํ•ด ์ดํ•ดํ•˜๊ธฐ

  • ์ค‘์•™ ์ง‘์ค‘์‹ ์ €์žฅ์†Œ๋ฅผ Store๋ผ๊ณ  ํ•ด๋ณด์ž.
  • Store์™€ Component์˜ ๊ด€๊ณ„
    • ํ•˜๋‚˜์˜ Store๊ฐ€ ์—ฌ๋Ÿฌ ๊ฐœ์˜ Component์—์„œ ์‚ฌ์šฉ๋  ์ˆ˜ ์žˆ๋‹ค.
    • Store๊ฐ€ ๋ณ€๊ฒฝ๋  ๋•Œ, Store๊ฐ€ ์‚ฌ์šฉ๋˜๊ณ  ์žˆ๋Š” Component๋„ ๋ณ€๊ฒฝ๋˜์–ด์•ผ ํ•œ๋‹ค.
// store์— ์ดˆ๊ธฐ state๋ฅผ ์ „๋‹ฌํ•˜๋ฉด์„œ ์ƒ์„ฑ
const store = new Store({
  a: 11,
  b: 16,
});

// ์ปดํฌ๋„ŒํŠธ 2๊ฐœ ์ƒ์„ฑ
// Q. ์—ฌ๊ธฐ์„œ Component์˜ parameter๋กœ ๊ฐ์ฒด๊ฐ€ ์ „๋‹ฌ๋˜๋Š” ๊ฒŒ ๋งž๋‚˜..? ์–ด์ฐจํ”ผ ์•„๋ž˜์—์„œ ๋”ฐ๋กœ ๊ตฌ๋…ํ•˜๋Š”๋ฐ?
const component1 = new Component({ subscribe: [store] });
const component2 = new Component({ subscribe: [store] });

// ์ปดํฌ๋„ŒํŠธ๊ฐ€ store๋ฅผ ๊ตฌ๋…
component1.subscribe(store); // a + b = ${store.state.a + store.state.b}
component2.subscribe(store); // a * b = ${store.state.a * store.state.b}

// store์˜ state๋ฅผ ๋ณ€๊ฒฝํ•œ๋‹ค.
store.setState({
  a: 7,
  b: 19,
});

// store๊ฐ€ ๋ณ€๊ฒฝ๋˜์—ˆ์Œ์„ ์•Œ๋ฆฐ๋‹ค.
store.notify();

์œ„์™€ ๊ฐ™์€ ํ˜•ํƒœ๋กœ ์ฝ”๋“œ๋ฅผ ์ž‘์„ฑํ•˜๋Š” ๊ฒƒ์„ Observer Pattern์ด๋ผ๊ณ  ํ•œ๋‹ค.

  • Observer Pattern์€ ๊ฐ์ฒด์˜ ์ƒํƒœ ๋ณ€ํ™”๋ฅผ ๊ด€์ฐฐํ•˜๋Š” ๊ด€์ฐฐ์ž๋“ค, ์ฆ‰ ์˜ต์ €๋ฒ„๋“ค์˜ ๋ชฉ๋ก์„ ๊ฐ์ฒด์— ๋“ฑ๋กํ•œ๋‹ค.
  • ์ƒํƒœ ๋ณ€ํ™”๊ฐ€ ์žˆ์„ ๋•Œ๋งˆ๋‹ค ๋ฉ”์„œ๋“œ ๋“ฑ์„ ํ†ตํ•ด ๊ฐ์ฒด๊ฐ€ ์ง์ ‘ ๋ชฉ๋ก์˜ ๊ฐ ์˜ต์ €๋ฒ„์—๊ฒŒ ํ†ต์ง€ํ•˜๋„๋ก ํ•œ๋‹ค.
  • ๋””์ž์ธ ํŒจํ„ด์˜ ํ•˜๋‚˜์ด๋‹ค.
  • ๋ฐœํ–‰/๊ตฌ๋… ๋ชจ๋ธ์ด๋ผ๊ณ ๋„ ๋ถ€๋ฅธ๋‹ค.

1. Publisher(๋ฐœํ–‰)

class Publisher {
  #state;
  #observers = new Set(); // ์—ฌ๊ธฐ์— ๊ตฌ๋…์ž๋“ค(๊ด€์ฐฐ์ž๋“ค)์ด ์ €์žฅ๋œ๋‹ค.
  
  constructor(state) {
    this.#state = state;
    // Object.defineProperty()
    // state ๊ฐ์ฒด์˜ key๊ฐ’์„ ์ˆœํšŒํ•˜๋ฉด์„œ this์— ํ•ด๋‹น key๊ฐ’์œผ๋กœ `() => this.#state[key]` value๋ฅผ ๊ฐ–๊ฒŒ ํ•œ๋‹ค.
    // Q. ๊ทธ๋Ÿฐ๋ฐ ์ด๊ฑธ ์™œํ•˜๋Š”๊ฑด์ง€ ๋ชจ๋ฅด๊ฒ ๋‹ค. 
    // - ์ƒ์„ฑ๋œ ์ธ์Šคํ„ด์Šค์—์„œ state๊ฐ€ ๊ฐ–๋Š” ๊ฐ’๋“ค์— ๋ฐ”๋กœ ์ ‘๊ทผํ•  ์ˆ˜ ์žˆ์–ด์„œ..?
    // A. ๊ฐ’ ๋ณ€๊ฒฝ์ด ์ผ์–ด๋‚˜์ง€ ์•Š๊ฒŒ ์ฒ˜๋ฆฌํ•˜๋Š” ๊ฒƒ ๊ฐ™๋‹ค.
    Object.keys(state).forEach(key => Object.defineProperty(this, key, {
      get: () => this.#state[key]
        }
      )
    );
  }
  
  setState(newState) {
    this.#state = {...this.#state, ...newState};
    this.notify();
  }
  
  register(observer) {
    this.#observers.add(observer);
  }
  // Q. notify์— ํŒŒ๋ผ๋ฏธํ„ฐ๊ฐ€ ์ „๋‹ฌ๋˜์ง€ ์•Š๋Š” ๋ถ€๋ถ„์ด ์ž˜ ์ดํ•ด๊ฐ€ ์•ˆ๋œ๋‹ค.
  notify() {
    this.#observers.forEach(observer => observer());
  }
}
  • Publisher: ๋ฐœํ–‰์ž
  • setState(): ๋ฐœํ–‰์ž์˜ ์ƒํƒœ๋ฅผ ๋ณ€๊ฒฝํ•˜๊ณ  ๊ตฌ๋…์ž๋ฅผ ์‹คํ–‰ํ•จ์œผ๋กœ์จ ์ƒํƒœ๋ณ€ํ™”๋ฅผ ์•Œ๋ฆฐ๋‹ค.
  • register(): ๊ตฌ๋…์ž(๊ด€์ฐฐ์ž)๋ฅผ ๋“ฑ๋กํ•œ๋‹ค.
  • notify(): ๊ตฌ๋…์ž(๊ด€์ฐฐ์ž; ํ•จ์ˆ˜)๋ฅผ ์‹คํ–‰ํ•œ๋‹ค.

์—ฌ๊ธฐ์„œ ํ•ต์‹ฌ์€ setState()๋ฅผ ํ†ตํ•ด ์ƒํƒœ ๋ณ€๊ฒฝ ์‹œ, notify()๊ฐ€ ์‹คํ–‰๋˜๋ฉฐ ๊ตฌ๋…์ž๋“ค์ด ์‹คํ–‰๋œ๋‹ค๋Š” ์ ์ด๋‹ค. ์ฆ‰, ๋‚ด๋ถ€ ์ƒํƒœ๊ฐ€ ๋ณ€ํ•  ๋•Œ ๊ตฌ๋…์ž์—๊ฒŒ ์•Œ๋ฆฌ๋Š” ๊ฒƒ

2. Subscriber(๊ตฌ๋…)

class Subscriber {
  #fn;
  constructor(fn) {
    this.#fn = fn;
  }

  subscribe(publisher) {
    publisher.register(this.#fn);
  }
}
  • Subscriber: ๊ตฌ๋…์ž
  • #fn: Publisher ์ž…์žฅ์—์„œ observer ๋œ๋‹ค.
  • subscribe(): publisher๋ฅผ ๋ฐ›์•„์„œ observer๋ฅผ ๋“ฑ๋กํ•œ๋‹ค.

๋ฐœํ–‰๊ธฐ๊ด€์„ ๊ตฌ๋…ํ•œ๋‹ค.
๋ฐœํ–‰๊ธฐ๊ด€์—์„œ ๋ณ€ํ™”๊ฐ€ ์ƒ๊ฒผ์„ ๋•Œ ํ•˜๋Š” ์ผ์„ ์ •์˜ํ•ด์•ผ ํ•œ๋‹ค. ์ด๊ฒŒ ์‚ฌ์‹ค publisher์—๊ฒŒ๋Š” observer๊ฐ€ ๋œ๋‹ค.

3. ์ ์šฉํ•˜๊ธฐ

const initailState = {a: 10, b: 20};

// Q. ์™œ publisher๋ผ๊ณ  ์•ˆํ•˜๊ณ  ์ƒํƒœ๋ผ๊ณ  ํ–ˆ์„๊นŒ..?
const publisher = new Publisher(initailState);

// Q-a. publisher๋Š” ์™ธ๋ถ€ ๊ฐ์ฒด์ธ๋ฐ, ์ด๊ฑธ ์ฐธ์กฐํ•˜๊ณ  ์žˆ๋Š” ๊ฒŒ ๋งž๋‚˜..?
const addCalculator = new Subscriber(() => console.log(`a + b = ${publisher.a + publisher.b}`));

// Q-b. publisher๋ฅผ ๊ตฌ๋…ํ•˜๋Š” ๊ฑด ์—ฌ๊ธฐ์„œ ํ•˜๋ฉด์„œ..? 
addCalculator.subscribe(publisher);

3. ๋ฆฌํŒฉํ† ๋ง

์•ž์˜ ์ฝ”๋“œ๋ฅผ ๋‹จ์ˆœํ•˜๊ฒŒ observable๊ณผ observe์˜ ๊ด€๊ณ„์—๋งŒ ์ง‘์ค‘ํ•ด์„œ ๋‹ค๋ค„๋ณด์ž.

  • observable์€ observe์—์„œ ์‚ฌ์šฉ๋œ๋‹ค.
  • observable์— ๋ณ€ํ™”๊ฐ€ ์ƒ๊ธฐ๋ฉด, observe์— ๋“ฑ๋ก๋œ ํ•จ์ˆ˜๊ฐ€ ์‹คํ–‰๋œ๋‹ค.

1. Object.defineProperty ์ดํ•ดํ•˜๊ธฐ

MDN ๋ฌธ์„œ์˜ ์„ค๋ช…์€ ์•„๋ž˜์™€ ๊ฐ™๋‹ค.

๊ฐ์ฒด์— ์ง์ ‘ ์ƒˆ๋กœ์šด ์†์„ฑ์„ ์ •์˜ํ•˜๊ฑฐ๋‚˜ ์ด๋ฏธ ์กด์žฌํ•˜๋Š” ์†์„ฑ์„ ์ˆ˜์ •ํ•œ ํ›„, ๊ทธ ๊ฐ์ฒด๋ฅผ ๋ฐ˜ํ™˜ํ•ฉ๋‹ˆ๋‹ค.

// ์—ฌ๊ธฐ์„œ ๋ถ€ํ„ฐ ์•„๋ž˜๊นŒ์ง€์˜ ์ฝ”๋“œ๋ฅผ class ํ•˜๋‚˜๋ผ๊ณ  ์ƒ๊ฐํ•˜๋ฉด ์ดํ•ดํ•˜๊ธฐ ํŽธํ•˜๋‹ค.
let a = 10;

const state = {};

Object.defineProperty(state, 'a', {
  get() {
    console.log(`ํ˜„์žฌ a์˜ ๊ฐ’์€ ${a} ์ž…๋‹ˆ๋‹ค.`);
    return a;
  },
  set(value) {
    a = value;
    console.log(`๋ณ€๊ฒฝ๋œ a์˜ ๊ฐ’์€ ${a} ์ž…๋‹ˆ๋‹ค.`)
  }
})
  • Object.defineProperty(targetObject, property, descriptor)
    • object: ์†์„ฑ์„ ์ •์˜ํ•  ๊ฐ์ฒด
    • property: ์ƒˆ๋กœ ์ •์˜ํ•˜๊ฑฐ๋‚˜ ์ˆ˜์ •ํ•˜๋ ค๋Š” ์†์„ฑ์˜ ์ด๋ฆ„ ๋˜๋Š” Symbol
    • descriptor: ์ƒˆ๋กœ ์ •์˜ํ•˜๊ฑฐ๋‚˜ ์ˆ˜์ •ํ•˜๋ ค๋Š” ์†์„ฑ์„ ๊ธฐ์ˆ ํ•˜๋Š” ๊ฐ์ฒด

Object.defineProperty(targetObject, property, descriptor)๋Š” ๊ฐ์ฒด๋ฅผ ์ฐธ์กฐํ•˜๊ฑฐ๋‚˜ ๊ฐ์ฒด์— ์–ด๋–ค ๋ณ€ํ™”๊ฐ€ ์ƒ๊ธธ ๋•Œ, ์ค‘๊ฐ„์— ์šฐ๋ฆฌ๊ฐ€ ์›ํ•˜๋Š” ํ–‰์œ„๋ฅผ ์ง‘์–ด๋„ฃ์„ ์ˆ˜ ์žˆ๊ฒŒ ํ•ด์ค€๋‹ค.

2. ์—ฌ๋Ÿฌ ๊ฐœ์˜ ์†์„ฑ ๊ด€๋ฆฌํ•˜๊ธฐ

const state = {
  a: 10,
  b: 20,
};

const stateKeys = Object.keys(state);

for (const key of stateKeys) {
  let _value = state[key];
  Object.defineProperty(state, key, {
    get () {
      console.log(`ํ˜„์žฌ state.${key}์˜ ๊ฐ’์€ ${_value} ์ž…๋‹ˆ๋‹ค.`);
      return _value;
    },
    set (value) {
      _value = value;
      console.log(`๋ณ€๊ฒฝ๋œ state.${key}์˜ ๊ฐ’์€ ${_value} ์ž…๋‹ˆ๋‹ค.`);
    }
  })
}

console.log(`a + b = ${state.a + state.b}`);

state.a = 100;
state.b = 200;

์œ„์˜ ์ฝ”๋“œ์—์„œ console.log๋งŒ observer๋ผ๋Š” ํ•จ์ˆ˜๋กœ ๋ณ€๊ฒฝํ•ด๋ณด์ž.

const state = {
  a: 10,
  b: 20,
};

const stateKeys = Object.keys(state);
const observer = () => console.log(`a + b = ${state.a + state.b}`);

for (const key of stateKeys) {
  let _value = state[key];
  Object.defineProperty(state, key, {
    get () {
      return _value;
    },
    set (value) {
      _value = value;
      observer(); // setํ•  ๋•Œ ์–ด๋–ค ํ•จ์ˆ˜๋ฅผ ์‹คํ–‰ํ•˜๊ฒŒ ํ•˜๋Š” ๊ฒƒ์ด๋‹ค.
    }
  })
}

observer();

state.a = 100;
state.b = 200;

์ง€๊ธˆ๊นŒ์ง€์˜ ๊ณผ์ •์„ ์กฐ๊ธˆ ์‰ฝ๊ฒŒ ์„ค๋ช…ํ•˜๋ฉด ๋‹ค์Œ๊ณผ ๊ฐ™์€ ํ๋ฆ„์ด๋‹ค.

  • obj.a๋กœ ์–ด๋–ค ๊ฐ’์— ์ ‘๊ทผํ•  ๋•Œ(get), ์ค‘๊ฐ„์— ๋ญ”๊ฐ€ ์–ด๋–ค ์ฝ”๋“œ๋ฅผ ์‹คํ–‰ํ•˜๊ณ  ์‹ถ์€ ๊ฒƒ. ์˜ต์ €๋ฒ„ ํŒจํ„ด์—์„œ๋Š” ์ด ์ค‘๊ฐ„์— ๊ตฌ๋…ํ•˜๋Š” ๋ฉ”์„œ๋“œ(subscribe)๊ฐ€ ๋“ค์–ด๊ฐ„๋‹ค.
  • obj.a์— ์–ด๋–ค ๊ฐ’์„ ํ• ๋‹นํ•  ๋•Œ(set), ์ค‘๊ฐ„์— ๋ญ”๊ฐ€ ์–ด๋–ค ์ฝ”๋“œ๋ฅผ ์‹คํ–‰ํ•˜๊ณ  ์‹ถ์€ ๊ฒƒ. ์˜ต์ €๋ฒ„ ํŒจํ„ด์—์„œ๋Š” ์ด ์ค‘๊ฐ„์— ์•Œ๋ฆผํ•˜๋Š” ๋ฉ”์„œ๋“œ(notify)๊ฐ€ ๋“ค์–ด๊ฐ„๋‹ค.
  • ๋งˆ์น˜ ์ค‘๊ฐ„์— ๊ณผ์ •์„ ๊ฐ€๋กœ์ฑ„์„œ ๋ฌด์–ธ๊ฐ€๋ฅผ ์‹คํ–‰ํ•˜๋Š” ๊ฒƒ. Proxy ํŒจํ„ด๊ณผ ์œ ์‚ฌํ•˜๋‹ค.

3. ์—ฌ๋Ÿฌ ๊ฐœ์˜ Observer ๊ด€๋ฆฌํ•˜๊ธฐ

let currentObserver = null;

const state = {
  a: 10,
  b: 20,
};

const stateKeys = Object.keys(state);

for (const key of stateKeys) {
  let _value = state[key];
  // Q. ์•„๋ž˜ observers๋Š” for ๋ฌธ ๋ฐ–์—์„œ ์„ ์–ธํ•ด๋„ ๋˜๋Š”๋ฐ, ๊ตณ์ด ์—ฌ๊ธฐ์— ํ•œ ์ด์œ ๊ฐ€ ์žˆ์„๊นŒ?
  // A. for๋ฌธ ์•ˆ์—์„œ ๊ฐ๊ฐ์˜ key์— ๋Œ€ํ•œ closer๋กœ ๊ฐ–๊ณ  ์žˆ์–ด์•ผ ๊ฐ key๊ฐ€ get ๋์„ ๋•Œ, ๋ณธ์ธ์—๊ฒŒ ํ•ด๋‹นํ•˜๋Š” ํ•จ์ˆ˜์—๋งŒ ์ ‘๊ทผํ•œ๋‹ค.
  // ๋งŒ์•ฝ ๋ฐ–์— ๋นผ๋‘๋ฉด ๋ชจ๋“  key๊ฐ’๋“ค์— ๋Œ€ํ•œ observers๊ฐ€ ๊ณต์œ ๋˜๋ฏ€๋กœ ๊ทธ ์•ˆ์— ๋ชจ๋“  ํ•จ์ˆ˜๊ฐ€ ์‹คํ–‰๋˜์–ด๋ฒ„๋ฆฐ๋‹ค.
  const observers = new Set();
  Object.defineProperty(state, key, {
    get () {
      if (currentObserver) observers.add(currentObserver);
      return _value;
    },
    set (value) {
      _value = value;
      observers.forEach(observer => observer());
    }
  })
}

const ๋ง์…ˆ_๊ณ„์‚ฐ๊ธฐ = () => {
  currentObserver = ๋ง์…ˆ_๊ณ„์‚ฐ๊ธฐ;
  // ์•„๋ž˜์—์„œ state.a๋ฅผ getํ•˜๋Š” ์ˆœ๊ฐ„ defineProperty์˜ get ๋ฉ”์„œ๋“œ๊ฐ€ ์‹คํ–‰๋œ๋‹ค.
  console.log(`a + b = ${state.a + state.b}`);
}

const ๋บ„์…ˆ_๊ณ„์‚ฐ๊ธฐ = () => {
  currentObserver = ๋บ„์…ˆ_๊ณ„์‚ฐ๊ธฐ;
  console.log(`a - b = ${state.a - state.b}`);
}

๋ง์…ˆ_๊ณ„์‚ฐ๊ธฐ();
// ์•„๋ž˜์—์„œ state.a์— ์–ด๋–ค ๊ฐ’์„ ํ• ๋‹นํ•˜๋Š” ์ˆœ๊ฐ„ defineProperty์˜ set ๋ฉ”์„œ๋“œ๊ฐ€ ์‹คํ–‰๋œ๋‹ค.
state.a = 100;

๋บ„์…ˆ_๊ณ„์‚ฐ๊ธฐ();
state.b = 200;

state.a = 1;
state.b = 2;

ํ•จ์ˆ˜๊ฐ€ ์‹คํ–‰๋  ๋•Œ currentObserver๊ฐ€ ์‹คํ–‰์ค‘์ธ ํ•จ์ˆ˜๋ฅผ ์ฐธ์กฐํ•˜๋„๋ก ๋งŒ๋“ ๋‹ค.
state์˜ ํ”„๋กœํผํ‹ฐ๊ฐ€ ์‚ฌ์šฉ๋  ๋•Œ(get์„ ํ˜ธ์ถœํ•  ๋•Œ),

4. ํ•จ์ˆ˜ํ™”

์œ„์˜ ์ฝ”๋“œ๋ฅผ ์žฌ์‚ฌ์šฉํ•˜๊ธฐ ์œ„ํ•ด onserve์™€ observable ํ•จ์ˆ˜๋กœ ๊ตฌํ˜„ํ•ด๋ณด์ž.

let currentObserver = null;

const observe = fn => {
  currentObserver = fn;
  // ์•„๋ž˜์—์„œ fn์ด ์‹คํ–‰๋  ๋•Œ, currentObserver๊ฐ€ ์กด์žฌํ•˜๊ฒŒ ๋˜๋ฏ€๋กœ observers์— fn์ด ๋“ฑ๋ก๋œ๋‹ค.
  fn();
  currentObserver = null;
}

const observable = obj => {
  Object.keys(obj).forEach(key => {
    let _value = obj[key];
    const observers = new Set();

    Object.defineProperty(obj, key, {
      get () {
        // get์ผ ๋•Œ๋Š” observers์— currentObserver๊ฐ€ ์ถ”๊ฐ€๋˜๊ธฐ๋งŒ ํ•œ๋‹ค.
        if (currentObserver) observers.add(currentObserver);
        return _value;
      },

      set (value) {
        _value = value;
        // set์ผ ๋•Œ๋Š” observers์— ์žˆ๋Š” currentObserver๋“ค์ด ํ•˜๋‚˜์”ฉ ํ˜ธ์ถœ๋œ๋‹ค.
        observers.forEach(fn => fn());
      }
    })
  })
  return obj;
}

์•„๋ž˜์™€ ๊ฐ™์ด ์‚ฌ์šฉํ•œ๋‹ค.

// Q. ์•„๋ž˜ ์ฝ”๋“œ ๋‹ค์‹œ ํ•œ๋ฒˆ ์ƒ๊ฐํ•ด๋ณด๊ธฐ
const ์ƒํƒœ = observable({ a: 10, b: 20 });
observe(() => console.log(`a = ${์ƒํƒœ.a}`));
observe(() => console.log(`b = ${์ƒํƒœ.b}`));
observe(() => console.log(`a + b = ${์ƒํƒœ.a} + ${์ƒํƒœ.b}`));
observe(() => console.log(`a * b = ${์ƒํƒœ.a} + ${์ƒํƒœ.b}`));
observe(() => console.log(`a - b = ${์ƒํƒœ.a} + ${์ƒํƒœ.b}`));

์ƒํƒœ.a = 100;
์ƒํƒœ.b = 200;

4. DOM์— ์ ์šฉํ•˜๊ธฐ

1. ์ผ๋‹จ ๊ตฌํ˜„ํ•ด๋ณด๊ธฐ

๊ตฌ์กฐ๋Š” ๋’ค๋กœ ํ•˜๊ณ , ๊ธฐ๋Šฅ๋งŒ ๊ตฌํ˜„ํ•ด๋ณด๊ธฐ.

  • index.html
<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="UTF-8">
	<title>Store๋ฅผ ์ ์šฉํ•ด๋ณด์ž</title>
</head>
<body>
	<div id="app"></div>
  <script type="module" src="./src/main.js"></script>
</body>
</html>
  • src/main.js
import { observable, observe } from "./core/observer.js";

const state = observable({
  a: 10,
  b: 20,
});

const $app = document.querySelector('#app');

const render = () => {
  $app.innerHTML = `
    <p>a + b = ${state.a + state.b}</p>
    <input id="stateA" value="${state.a}" />
    <input id="stateB" value="${state.b}" />
  `;

  $app.querySelector('#stateA').addEventListener('change', ({ target }) => {
    state.a = Number(target.value);
  })

  $app.querySelector('#stateB').addEventListener('change', ({ target }) => {
    state.b = Number(target.value);
  })
}

/* observe ํ•จ์ˆ˜๊ฐ€ ํ˜ธ์ถœ๋˜๋ฉด render๋„ ํ˜ธ์ถœ๋˜๋ฉด์„œ ๊ทธ ์•ˆ์— ์žˆ๋Š” state.a์™€ state.b๊ฐ€ getter๋กœ ํ˜ธ์ถœ๋  ๋•Œ
render ํ•จ์ˆ˜๊ฐ€ ๊ฐ๊ฐ์˜ observers์— ๋“ฑ๋ก๋œ๋‹ค.
๊ทธ๋ฆฌ๊ณ  #stateA์™€ #stateB์— ์–ด๋–ค ๊ฐ’์„ ์ž…๋ ฅํ•˜๊ณ  ๋ณ€ํ™”๋ฅผ ๊ฐ์ง€ํ•˜๋ฉด ๋ฐ”๋กœ state.a์™€ state.b์˜ setter๊ฐ€ ํ˜ธ์ถœ๋˜๋ฉด์„œ
๊ทธ ์•ˆ์— observer์ค‘ ํ•˜๋‚˜๋กœ ์กด์žฌํ•˜๋Š” render ํ•จ์ˆ˜๊ฐ€ ์‹คํ–‰๋˜๋ฉด์„œ ui๋ฅผ ๋ณ€๊ฒฝํ•œ๋‹ค.
 */
observe(render);

2. Component๋กœ ์ถ”์ƒํ™”ํ•˜๊ธฐ

  • src/core/Component.js
import { observable, observe } from './observer.js';

export class Component {
  state; 
  props; 
  $el;

  constructor ($el, props) {
    // $el์ด ๋ถ€๋ชจ ์—˜๋ฆฌ๋จผํŠธ์ด๋‹ค. ํ•ด๋‹น ์ปดํฌ๋„ŒํŠธ๊ฐ€ ์ถ”๊ฐ€๋  target
    this.$el = $el;
    this.props = props;
    this.setup();
  }

  setup() {
    // ์ปดํฌ๋„ŒํŠธ๋งˆ๋‹ค store๊ฐ€ ์žˆ๋Š” ๋Š๋‚Œ์ด๋„ค. ์—ฌ๊ธฐ์„œ state๋Š” store๋ผ๊ณ  ํ•˜๋Š” ๊ฒŒ ์ข€๋” ์ž์—ฐ์Šค๋Ÿฝ์ง€ ์•Š๋‚˜ ์‹ถ๋‹ค.
    this.state = observable(this.initState()); // state๋ฅผ ๊ด€์ฐฐํ•œ๋‹ค.
    observe(() => { // state๊ฐ€ ๋ณ€๊ฒฝ๋  ๊ฒฝ์šฐ, ํ•จ์ˆ˜๊ฐ€ ์‹คํ–‰๋œ๋‹ค.
      this.render();
      this.setEvent();
      this.mounted();
    });
  }

  initState() { return {} }
  template () { return ''; }
  render () { this.$el.innerHTML = this.template(); }
  setEvent () {}
  mounted () {}
}

์œ„ ์ฝ”๋“œ๋ฅผ ๊ธฐ๋ฐ˜์œผ๋กœ src/App.js์— Component๋ฅผ ์ ์šฉํ•ด๋ณด์ž.

import {Component} from "./core/Component.js";

export class App extends Component {
  // constructor์™€ super๋ฅผ ์•Œ์•„์„œ ๋„ฃ์–ด์ค€๋‹ค.
  initState () {
    return {
      a: 10,
      b: 20,
    }
  }

  template () {
    // ์ด ๋•Œ, this.state๋Š” ์ด๋ฏธ observable์ด ์ ์šฉ๋œ ๊ฐ์ฒด์ด๋‹ค.
    const { a, b } = this.state;
    return `
      <input id="stateA" value="${a}" size="5" />
      <input id="stateB" value="${b}" size="5" />
      <p>a + b = ${a + b}</p>
    `;
  }

  setEvent () {
    const { $el, state } = this;

    $el.querySelector('#stateA').addEventListener('change', ({ target }) => {
      // state.a์— ๊ฐ’์ผ ํ• ๋‹น๋  ๋•Œ(setter), `render, setEvent, mounted`๊ฐ€ ์‹คํ–‰๋œ๋‹ค.
      state.a = Number(target.value);
    })

    $el.querySelector('#stateB').addEventListener('change', ({ target }) => {
      state.b = Number(target.value);
    })
  }
}

3. ๊ณ ๋ฏผํ•ด๋ณด๊ธฐ

์ด๋ ‡๊ฒŒ Component ๋‚ด๋ถ€์—์„œ ๊ด€๋ฆฌ๋˜๋Š” state(store ๋Š๋‚Œโ€ฆ!)์— observable์„ ์”Œ์›Œ์„œ ์‚ฌ์šฉํ•  ๊ฒฝ์šฐ, setState๋ฅผ ์‚ฌ์šฉํ•˜๋Š” ๋ฐฉ์‹๊ณผ ํฌ๊ฒŒ ๋‹ค๋ฅด์ง€ ์•Š๋‹ค๊ณ  ๋Š๋‚„ ์ˆ˜ ์žˆ๋‹ค. setState() ๋˜ํ•œ state๊ฐ€ ๋ณ€๊ฒฝ๋  ๋•Œ๋งˆ๋‹ค render๋ฅผ ์‹คํ–‰ํ•˜๋Š” ๋ฐฉ์‹์ด๊ธฐ ๋•Œ๋ฌธ์ด๋‹ค.(์ง์ ‘ ์‚ฌ์šฉํ•˜๋Š๋ƒ, setter๋ฅผ ํ–ˆ์„ ๋•Œ ์‹คํ–‰ํ•˜๋Š๋ƒ์˜ ์ฐจ์ด ๋Š๋‚Œ)

observer๋Š” ์ด๋ ‡๊ฒŒ ์ปดํฌ๋„ŒํŠธ ๋‚ด๋ถ€์—์„œ ์‚ฌ์šฉํ•˜๊ธฐ๋ณด๋‹จ ์ค‘์•™ ์ง‘์ค‘์‹ ์ €์žฅ์†Œ๋ฅผ ๊ด€๋ฆฌํ•  ๋•Œ ๋งค์šฐ ํšจ๊ณผ์ ์ด๋‹ค!!!

4. ์ปดํฌ๋„ŒํŠธ ์™ธ๋ถ€์— ์ƒํƒœ ๋งŒ๋“ค์–ด์ฃผ๊ธฐ

์•„์ฃผ ๊ฐ„๋‹จํ•œ Store๋ฅผ ๋งŒ๋“ค์–ด์„œ ๊ด€๋ฆฌํ•ด๋ณด์ž.

  • src/store.js
import { observable } from './core/observer.js'

export const store = {
  state: observable({
    a: 10,
    b: 20,
  }),

  setState (newState) {
    for (const [key, value] of Object.entries(newState)) {
      if (!this.state[key]) continue;
      this.state[key] = value;
    }
  }
}
  • src/App.js
import { Component } from "./core/Component.js";
import { store } from './store.js';

const InputA = () => `
  <input id="stateA" value="${store.state.a}" size="5" />
`;

const InputB = () => `
  <input id="stateB" value="${store.state.b}" size="5" />
`

const Calculator = () => `
  <p>a + b = ${store.state.a + store.state.b}</p>
`

export class App extends Component {
  template () {
    return `
      ${InputA()}
      ${InputB()}
      ${Calculator()}
    `;
  }

  setEvent () {
    const { $el } = this;

    $el.querySelector('#stateA').addEventListener('change', ({ target }) => {
      store.setState({ a: Number(target.value) });
    })

    $el.querySelector('#stateB').addEventListener('change', ({ target }) => {
      store.setState({ b: Number(target.value) });
    })
  }
}
  • InputA, InputB, Calculator 3๊ฐœ์˜ ์ปดํฌ๋„ŒํŠธ๊ฐ€ store๋ฅผ ์ฐธ์กฐํ•˜๊ณ  ์žˆ๊ณ , store๊ฐ€ ๋ณ€๊ฒฝ๋˜์—ˆ์„ ๋•Œ ์ปดํฌ๋„ŒํŠธ๊ฐ€ ์ž๋™์œผ๋กœ ๋ Œ๋”๋ง๋œ๋‹ค.
  • ์—ฌ๊ธฐ์— Flux ํŒจํ„ด์„ ์‚ฌ์šฉํ•˜๋ฉด Redux๋‚˜ Vuex๊ฐ€ ๋˜๋Š” ๊ฒƒ์ด๋‹ค.

5. Flux Pattern

  • Redux๋‚˜ Vuex์—์„œ ์‚ฌ์šฉ๋˜๋Š” Flux ํŒจํ„ด
  • ๋‹จ๋ฐฉํ–ฅ ๋ฐ์ดํ„ฐ ํ๋ฆ„
    • Dispatcher -> Store
    • Store -> View
    • View -> Action
    • Action -> Dispatcher
  • ๋‹จ๋ฐฉํ–ฅ ๋ฐ์ดํ„ฐ ํ๋ฆ„์€ ๋ฐ์ดํ„ฐ์˜ ๋ณ€ํ™”๋ฅผ ํ›จ์”ฌ ์˜ˆ์ธกํ•˜๊ธฐ ์‰ฝ๊ฒŒ ๋งŒ๋“ ๋‹ค.

๋˜ํ•œ, ์šฐ๋ฆฌ๊ฐ€ ๋ˆˆ์œผ๋กœ ๋ณด๋Š” UI์— ๋”ฐ๋ผ ์ผ๋Œ€์ผ ๋Œ€์‘์œผ๋กœ state์˜ ๋ณ€ํ™”๋ฅผ ํ™•์ธํ•  ์ˆ˜ ์žˆ๋‹ค๋ฉด ์–ผ๋งˆ๋‚˜ ์ข‹๊ฒ ๋Š”๊ฐ€?!

6. Vuex ๊ฐ™์€ Store ๋งŒ๋“ค๊ธฐ

Vuex๊ฐ€ ์–ด๋–ค interface๋ฅผ ๊ฐ€์ง€๊ณ  ์žˆ๋Š”์ง€ ๊ฐ„๋‹จํ•˜๊ฒŒ ์‚ดํŽด๋ณด์ž.

const store = new Vuex.Store({
  state: {
    count: 0
  },
  mutations: {
    increment (state) {
      state.count++
    }
  }
});
  • ์ฐธ๊ณ : Redux reducer๊ณผ Vuex mutations์˜ ์ฐจ์ด์— ๋Œ€ํ•œ GPT์˜ ๋‹ต๋ณ€

์งˆ๋ฌธ: redux reducer and vuex mutations are simillar, right?
๋‹ต๋ณ€: Yes, Redux reducers and Vuex mutations are similar in their purpose and functionality. Both Redux and Vuex are state management libraries used in JavaScript applications, and reducers/mutations are used to modify the state in a predictable way. Redux is commonly used in React applications, while Vuex is used in Vue.js applications. Both Redux reducers and Vuex mutations take the current state and an action/payload as input, and return a new state as output. The new state is based on the current state and the action/payload passed to the reducer/mutation. The main difference between the two is in how they are defined and used. In Redux, reducers are pure functions that take the current state and an action, and return a new state. Reducers are combined together using the combineReducers function, and are called by the Redux store when an action is dispatched. In Vuex, mutations are functions that directly modify the state. They are defined as methods in the Vuex store, and are called using the commit method when an action is dispatched. Overall, while there are some differences in their implementation, Redux reducers and Vuex mutations serve a similar purpose and are used to achieve similar goals in state management.

  • mutations: state๋ฅผ ๋ณ€๊ฒฝ์‹œํ‚จ๋‹ค.
store.commit('increment'); // state์˜ count ๊ฐ’์ด 1 ์ฆ๊ฐ€ํ•˜๊ฒŒ ๋œ๋‹ค.

console.log(store.state.count); // 1

์ฆ‰, commit์œผ๋กœ mutations์˜ ๋ฉ”์„œ๋“œ๋ฅผ ์‹คํ–‰ํ•˜๋Š” ๊ตฌ์กฐ์ด๋‹ค.

Vuex๋Š” ๊ฐœ๋…๋งŒ ์•Œ์•„๋ณด๊ณ  ๊ตฌํ˜„์‚ฌํ•ญ์€ ์ƒ๋žตํ•˜๊ฒ ์Šต๋‹ˆ๋‹ค.

7. Redux ๋งŒ๋“ค๊ธฐ

๋งˆ์ฐฌ๊ฐ€์ง€๋กœ Redux๊ฐ€ ์‚ฌ์šฉ๋˜๋Š” ํ˜•ํƒœ๋ฅผ ์‚ดํŽด๋ณด์ž.

import { createStore } from 'redux'

/**
 * ์ด๊ฒƒ์ด (state, action) => state ํ˜•ํƒœ์˜ ์ˆœ์ˆ˜ ํ•จ์ˆ˜์ธ ๋ฆฌ๋“€์„œ์ž…๋‹ˆ๋‹ค.
 * ๋ฆฌ๋“€์„œ๋Š” ์•ก์…˜์ด ์–ด๋–ป๊ฒŒ ์ƒํƒœ๋ฅผ ๋‹ค์Œ ์ƒํƒœ๋กœ ๋ณ€๊ฒฝํ•˜๋Š”์ง€ ์„œ์ˆ ํ•ฉ๋‹ˆ๋‹ค.
 *
 * ์ƒํƒœ์˜ ๋ชจ์–‘์€ ๋‹น์‹  ๋งˆ์Œ๋Œ€๋กœ์ž…๋‹ˆ๋‹ค: ๊ธฐ๋ณธํ˜•(primitive)์ผ์ˆ˜๋„, ๋ฐฐ์—ด์ผ์ˆ˜๋„, ๊ฐ์ฒด์ผ์ˆ˜๋„,
 * ์‹ฌ์ง€์–ด Immutable.js ์ž๋ฃŒ๊ตฌ์กฐ์ผ์ˆ˜๋„ ์žˆ์Šต๋‹ˆ๋‹ค. ์˜ค์ง ์ค‘์š”ํ•œ ์ ์€ ์ƒํƒœ ๊ฐ์ฒด๋ฅผ ๋ณ€๊ฒฝํ•ด์„œ๋Š” ์•ˆ๋˜๋ฉฐ,
 * ์ƒํƒœ๊ฐ€ ๋ฐ”๋€๋‹ค๋ฉด ์ƒˆ๋กœ์šด ๊ฐ์ฒด๋ฅผ ๋ฐ˜ํ™˜ํ•ด์•ผ ํ•œ๋‹ค๋Š” ๊ฒƒ์ž…๋‹ˆ๋‹ค. => ๋ถˆ๋ณ€์„ฑ
 *
 * ์ด ์˜ˆ์ œ์—์„œ ์šฐ๋ฆฌ๋Š” `switch` ๊ตฌ๋ฌธ๊ณผ ๋ฌธ์ž์—ด์„ ์ผ์ง€๋งŒ,
 * ์—ฌ๋Ÿฌ๋ถ„์˜ ํ”„๋กœ์ ํŠธ์— ๋งž๊ฒŒ
 * (ํ•จ์ˆ˜ ๋งต ๊ฐ™์€) ๋‹ค๋ฅธ ์ปจ๋ฒค์…˜์„ ๋”ฐ๋ฅด์…”๋„ ์ข‹์Šต๋‹ˆ๋‹ค.
 */
function counterReducer(state = 0, action) {
  switch (action.type) {
    case 'INCREMENT':
      return state + 1
    case 'DECREMENT':
      return state - 1
    default:
      return state
  }
}

// ์•ฑ์˜ ์ƒํƒœ๋ฅผ ๋ณด๊ด€ํ•˜๋Š” Redux ์ €์žฅ์†Œ๋ฅผ ๋งŒ๋“ญ๋‹ˆ๋‹ค.
// API๋กœ๋Š” { subscribe, dispatch, getState }๊ฐ€ ์žˆ์Šต๋‹ˆ๋‹ค.
let store = createStore(counter)

// subscribe()๋ฅผ ์ด์šฉํ•ด ์ƒํƒœ ๋ณ€ํ™”์— ๋”ฐ๋ผ UI๊ฐ€ ๋ณ€๊ฒฝ๋˜๊ฒŒ ํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.
// ๋ณดํ†ต์€ subscribe()๋ฅผ ์ง์ ‘ ์‚ฌ์šฉํ•˜๊ธฐ๋ณด๋‹ค๋Š” ๋ทฐ ๋ฐ”์ธ๋”ฉ ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ(์˜ˆ๋ฅผ ๋“ค์–ด React Redux)๋ฅผ ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค.
// ํ•˜์ง€๋งŒ ํ˜„์žฌ ์ƒํƒœ๋ฅผ localStorage์— ์˜์†์ ์œผ๋กœ ์ €์žฅํ•  ๋•Œ๋„ ํŽธ๋ฆฌํ•ฉ๋‹ˆ๋‹ค.

store.subscribe(() => console.log(store.getState()))

// ๋‚ด๋ถ€ ์ƒํƒœ๋ฅผ ๋ณ€๊ฒฝํ•˜๋Š” ์œ ์ผํ•œ ๋ฐฉ๋ฒ•์€ ์•ก์…˜์„ ๋ณด๋‚ด๋Š” ๊ฒƒ๋ฟ์ž…๋‹ˆ๋‹ค.
// ์•ก์…˜์€ ์ง๋ ฌํ™”ํ• ์ˆ˜๋„, ๋กœ๊น…ํ• ์ˆ˜๋„, ์ €์žฅํ• ์ˆ˜๋„ ์žˆ์œผ๋ฉฐ ๋‚˜์ค‘์— ์žฌ์‹คํ–‰ํ• ์ˆ˜๋„ ์žˆ์Šต๋‹ˆ๋‹ค.
store.dispatch({ type: 'INCREMENT' })
// 1
store.dispatch({ type: 'INCREMENT' })
// 2
store.dispatch({ type: 'DECREMENT' })
// 1

์ฝ”๋“œ๋ฅผ ์‚ดํŽด๋ณด๋ฉด createStore๊ฐ€ subscribe, dispatch, getState ๋“ฑ์˜ ๋ฉ”์„œ๋“œ๋ฅผ ๊ฐ€์ง„ ๊ฐ์ฒด๋ฅผ ๋ฐ˜ํ™˜ํ•˜๋Š” ๊ฒƒ์„ ์•Œ ์ˆ˜ ์žˆ๋‹ค.

const createStore = (reducer) => {
  return {subscribe, dispatch, getState};
}

์œ„์˜ ๋‚ด์šฉ์„ ๊ธฐ๋ฐ˜์œผ๋กœ ๊ฐ„๋‹จํ•˜๊ฒŒ ๊ตฌํ˜„์„ ํ•ด๋ณด์ž.

import { observable } from './observer.js';

export const createStore = (reducer) => {

  // reducer๊ฐ€ ์‹คํ–‰๋  ๋•Œ ๋ฐ˜ํ™˜ํ•˜๋Š” ๊ฐ์ฒด(state)๋ฅผ observable๋กœ ๋งŒ๋“ค์–ด์•ผ ํ•œ๋‹ค.
  // Q. ์—ฌ๊ธฐ์„œ state๋ฅผ observableํ•˜๋Š” ๊ฒŒ ์˜๋ฏธ๊ฐ€ ์žˆ๋‚˜..???
  // observable์„ ์ ์šฉํ•œ initialState๋ฅผ ๋ฐ›๋Š” ๊ฑธ๋กœ ์ดํ•ดํ•˜๋ฉด ๋ ๊นŒ?
  // ์–ด๋–ค ๊ฒƒ๋„ observe ํ•˜์ง€ ์•Š๋Š”๋‹ค๋ฉด, observable์€ ์•„๋ฌด๋Ÿฐ ์˜๋ฏธ๊ฐ€ ์—†์ง€ ์•Š๋‚˜..?
  const state = observable(reducer());

  // getState๊ฐ€ ์‹ค์ œ state๋ฅผ ๋ฐ˜ํ™˜ํ•˜๋Š” ๊ฒƒ์ด ์•„๋‹ˆ๋ผ frozenState๋ฅผ ๋ฐ˜ํ™˜ํ•˜๋„๋ก ๋งŒ๋“ค์–ด์•ผ ํ•œ๋‹ค.
  // state์™€ ๋™์ผํ•œ ๋‚ด์šฉ์ด์ง€๋งŒ, getter๋งŒ ๋˜๋ฉด์„œ state === frozenState๊ฐ€ false
  const frozenState = {};
  Object.keys(state).forEach(key => {
    Object.defineProperty(frozenState, key, {
      get: () => state[key], // get๋งŒ ์ •์˜ํ•˜์—ฌ set์„ ํ•˜์ง€ ๋ชปํ•˜๋„๋ก ๋งŒ๋“œ๋Š” ๊ฒƒ์ด๋‹ค.
    })
  });

  // dispatch๋กœ๋งŒ state์˜ ๊ฐ’์„ ๋ณ€๊ฒฝํ•  ์ˆ˜ ์žˆ๋‹ค.
  const dispatch = (action) => {
    const newState = reducer(state, action);

    for (const [key, value] of Object.entries(newState)) {
      // state์˜ key๊ฐ€ ์•„๋‹ ๊ฒฝ์šฐ ๋ณ€๊ฒฝ์„ ์ƒ๋žตํ•œ๋‹ค.
      if (!state[key]) continue;
      state[key] = value;
    }
  }

  const getState = () => frozenState;

  // subscribe๋Š” observe๋กœ ๋Œ€์ฒดํ•œ๋‹ค.
  return { getState, dispatch };
}

์œ„์˜ createStore๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ store๋ฅผ ๋งŒ๋“ค์–ด๋ณด์ž.

src/store.js

import {createStore} from './core/Store.js';

// ์ดˆ๊ธฐ state์˜ ๊ฐ’์„ ์ •์˜ํ•ด์ค€๋‹ค.
const initState = {
  a: 10,
  b: 20,
};

// dispatch์—์„œ ์‚ฌ์šฉ๋  type๋“ค์„ ์ •์˜ํ•ด์ค€๋‹ค.
export const SET_A = 'SET_A';
export const SET_B = 'SET_B';

// reducer๋ฅผ ์ •์˜ํ•˜์—ฌ store์— ๋„˜๊ฒจ์ค€๋‹ค.
export const store = createStore((state = initState, action = {}) => {
  switch (action.type) {
    case 'SET_A' :
      return { ...state, a: action.payload }
    case 'SET_B' :
      return { ...state, b: action.payload }
    default:
      return state;
  }
});

/* ์•„๋ž˜์™€ ๊ฐ™์ด reducer ํ•จ์ˆ˜๋ฅผ ์ •์˜ํ•ด์„œ ์ „๋‹ฌํ•ด๋„ ์ข‹์„๋“ฏ
const reducer = (state, action) => {
  switch (action.type) {
    case 'SET_A' :
      return { ...state, a: action.payload }
    case 'SET_B' :
      return { ...state, b: action.payload }
    default:
      return state;
  }
}
 */

// reducer์—์„œ ์‚ฌ์šฉ๋  action์„ ์ •์˜ํ•ด์ค€๋‹ค.
export const setA = (payload) => ({ type: SET_A, payload });
export const setB = (payload) => ({ type: SET_B, payload });

์ด์ œ App์—์„œ Store๋ฅผ ์‚ฌ์šฉํ•ด๋ณด์ž.

src/App.js

import { Component } from "./core/Component.js";
import {setA, setB, store} from './store.js';

const InputA = () => `<input id="stateA" value="${store.getState().a}" size="5" />`;
const InputB = () => `<input id="stateB" value="${store.getState().b}" size="5" />`;
const Calculator = () => `<p>a + b = ${store.getState().a + store.getState().b}</p>`;

export class App extends Component {
  template () {
    return `
      ${InputA()}
      ${InputB()}
      ${Calculator()}
    `;
  }

  setEvent () {
    const { $el } = this;

    $el.querySelector('#stateA').addEventListener('change', ({ target }) => {
      // dispatch๋ฅผ ํ†ตํ•ด์„œ ๊ฐ’์„ ๋ณ€๊ฒฝ์‹œํ‚จ๋‹ค.(commit์ด๋ผ๊ณ  ์˜คํƒ€ ์žˆ์Œ)
      store.dispatch(setA(Number(target.value)));
    })

    $el.querySelector('#stateB').addEventListener('change', ({ target }) => {
      // dispatch๋ฅผ ํ†ตํ•ด์„œ ๊ฐ’์„ ๋ณ€๊ฒฝ์‹œํ‚จ๋‹ค.(commit์ด๋ผ๊ณ  ์˜คํƒ€ ์žˆ์Œ)
      store.dispatch(setB(Number(target.value)));
    })
  }
}

์ด๋ ‡๊ฒŒ ์œ„์™€ ๊ฐ™์ด ๊ฐ„๋‹จํ•œ redux๋ฅผ ๋งŒ๋“ค์–ด๋ณผ ์ˆ˜ ์žˆ๋‹ค.

์ฐธ๊ณ