Devahoy Logo
PublishedAt

React

[React.js] หัดใช้งาน Zustand เพื่อจัดการ State

[React.js] หัดใช้งาน Zustand เพื่อจัดการ State

พอดีว่าผมห่างหายกับการเขียน React.js ไปพักใหญ่ๆ คือแทบไม่ได้เขียนโค๊ด หรือทำเว็บเลย ตั้งแต่ปีที่แล้ว และเพิ่งกลับมารื้อฟื้นใหม่ ก็ต้นปี 2022 ที่ผ่านมา ก็ได้ไปเจอตัว State Management ตัวนึงที่ชื่อว่า Zustand เพราะ ตัว Mascot มันเท่ดี หลังจากลองอ่าน Docs คร่าวๆ ก็รู้สึกว่า มันใช้งานง่าย สะดวกดี

จริงๆ นอกจาก Zustand มันก็ยังมี State Management อีกหลายตัวครับ ไม่ว่าจะเป็น Jotai, Recoil, MobX, Flux, Redux หรือแม้แต่ React Context API ก็ตาม ทุกตัว ไม่มีตัวไหนดีสุด มีข้อดี ข้อเสียต่างกันไป อยากใช้ตัวไหน ก็ตามสะดวกครับ และโพสต์นี้ก็ไม่ได้มาเชียร์ หรือเปรียบเทียบนะครับ

สำหรับบทความนี้ จะมาสรุปการใช้งาน Zustand เบื้องต้น เล็กๆน้อยๆ ซึ่ง ตัวอย่าง ก็นำมาจาก README ของทาง Zustand นั่นแหละครับ

  • Zustand Docs - หน้าเว็บ Docs เนื้อหาเดียวกับใน Github

Install Zustand

ทำการติดตั้ง zustand ด้วย Yarn หรือ NPM แล้วแต่สะดวก

1
npm install zustand
2
# หรือ yarn add zustand

ถ้าเราดูจากเว็บของ zustand ใน Github จะเห็นว่ามี ตัวอย่างง่ายๆ แบบด้านล่างเลย คือ

Create a Store

สร้าง Store ขึ้นมา โดยใช้ function create มี state ชื่อ bears มีค่าเป็น 0 และมี increasePopulation และ removeAllBears ที่เป็น function ไว้ set ค่า state

1
import create from 'zustand'
2
3
const useStore = create((set) => ({
4
bears: 0,
5
increasePopulation: () => set((state) => ({ bears: state.bears + 1 })),
6
removeAllBears: () => set({ bears: 0 })
7
}))

ซึ่งถ้าใครอ่านแล้ว งงๆ ลองแยกเป็น 2 ส่วน ส่วน createStore และ ส่วน useStore แบบนี้ดู น่าจะเข้าใจมากขึ้น และไม่สับสนตรง callback function

1
import create from 'zustand'
2
3
// 1 - create get/set states.
4
const bearStore = (set) => ({
5
bears: 0,
6
increasePopulation: () => set((state) => ({ bears: state.bears + 1 })),
7
removeAllBears: () => set({ bears: 0 })
8
})
9
10
// 2 - create store.
11
const useStore = create(bearStore)

Component

ทีนี้เวลาเข้าถึง state ก็ใช้ useStore(fn) โดยที่ fn คือ (state) => state.bears ก็จะได้แบบตัวอย่าง

1
function BearCounter() {
2
const bears = useStore((state) => state.bears)
3
return <h1>{bears} around here ...</h1>
4
}

ส่วนการ set ค่า ผ่าน function increasePopulation ก็ใช้ useStore() เหมือนกัน

1
function Controls() {
2
const increasePopulation = useStore((state) => state.increasePopulation)
3
return <button onClick={increasePopulation}>one up</button>
4
}

จะเห็นว่า ตัวอย่างมีแค่นี้ครับ เรียบง่ายดี ถ้าจะเปรียบเทียบแบบ setState() ปกติ ก็น่าจะประมาณนี้

1
function Controls() {
2
const [bears, increasePopulation] = useState(0)
3
4
return <button onClick={() => increasePopulation(bears + 1)}>one up</button>
5
}
1
const state = useStore()

แต่ข้อเสียคือ มันจะ render component ทุกๆครั้งที่ state เปลี่ยน ฉะนั้น เลือกเฉพาะ state ที่เราต้องการ

1
const myState = useState((state) => state.myState)

หลังจากอ่านไปแล้ว จะให้ดีที่สุด ก็ต้องลงมือทำเพื่อจะได้เห็นภาพ และเข้าใจมันมากขึ้น นั่นเอง

Create New Project

ลองสร้างโปรเจ็คขึ้นมาแบบเร็วๆ ผมเลือกใช้ Vite และเลือก React แบบ JavaScript ธรรมดา (หรือ TypeScript ขึ้นอยู่กับความถนัด)

Terminal window
yarn create vite
Project name: hello-zustand
Select a framework: react
Select a variant: react

ติดตั้ง zustand

1
yarn add zustand

ลองรัน Server ขึ้นมา

1
yarn dev

จะได้หน้าเว็บง่ายๆ ขึ้นมา ซึ่งสามารถกด button เพื่อเพิ่มค่า count ได้ ทีนี้ ก็เปลี่ยนจาก Default ที่เป็น useState มาใช้ zustand และตัว counter ต้องทำงานเหมือนเดิม

store.js
1
import create from 'zustand'
2
3
const useStore = create((set) => ({
4
count: 0,
5
setCount: () => set((state) => ({ count: state.count + 1 }))
6
}))
7
8
export default useStore

ลอง import ไฟล์มาใช้ใน App.jsx และเปลี่ยนเป็น useStore

App.jsx
1
import useStore from "./store"
2
3
function App() {
4
// const [count, setCount] = useState(0)
5
const count = useStore((state) => state.count)
6
const setCount = useStore((state) => state.setCount)
7
8
return (...)
9
}

แต่ถ้าแค่ state แบบนี้ จริงๆ ไม่จำเป็นต้องใช้ zustand เลย แค่ useState ก็เพียงพอแล้ว จริงมั้ย 🤣

ลองคิดกรณีสมมติขึ้นมีดีกว่า เช่น ต้องมีการ ให้เก็บ State เป็น Global State เพื่อที่จะใช้ร่วมกันในหลายๆ Components หรืออย่างเช่น Parent Component กับ Child Component จะส่งค่า get/set ค่ากันยังไงได้บ้าง

สร้าง store ขึ้นมาใหม่ ใน store.js ผมตั้งชื่อมันว่า useCounterStore

store.js
1
import create from 'zustand'
2
3
const counterStore = (set) => ({
4
count: 0,
5
increment: () => set((state) => ({ count: state.count + 1 })),
6
decrement: () => set((state) => ({ count: state.count - 1 }))
7
})
8
9
export const useCounterStore = create(counterStore)

เลยลองสร้าง component ใหม่ ชื่อ Counter.jsx ในโฟลเดอร์ components

components/Counter.jsx
1
import React from 'react'
2
import { useCounterStore } from '../store'
3
4
const Counter = () => {
5
const { count, increment, decrement } = useCounterStore()
6
7
console.log(`Counter:render`)
8
return (
9
<section>
10
<h3>Counter</h3>
11
<p>count : {count}</p>
12
<button onClick={increment}>เพิ่ม</button>
13
<button onClick={decrement}>ลบ</button>
14
</section>
15
)
16
}
17
18
export default Counter

จากนั้น ใน App.jsx ก็ import <Counter /> มาใช้ โดยเป้าหมายคือ ตัว App.jsx จะเรียก function เพื่อทำการ setState และ getState ที่เป็น global state ร่วมกันกับ Counter

App.jsx
1
import logo from './logo.svg'
2
import './App.css'
3
import { useCounterStore } from './store'
4
import Counter from './components/Counter'
5
6
function App() {
7
const { increment, decrement } = useCounterStore()
8
9
console.log(`App:render`)
10
11
return (
12
<div className="App">
13
<header className="App-header">
14
<img src={logo} className="App-logo" alt="logo" />
15
<p>Hello Vite + React!</p>
16
<button type="button" onClick={increment}>
17
Increment
18
</button>
19
<button type="button" onClick={decrement}>
20
Decrement
21
</button>
22
23
<Counter />
24
</header>
25
</div>
26
)
27
}
28
29
export default App

ทดลองเปิดหน้าเว็บ แล้วลองเปิด Dev Console ขึ้นมา เพื่อดู log เมื่อเรากด ปุ่ม เพิ่ม หรือ ลบ จริงๆ มันควรจะ render แค่ Component Counter ใช่มั้ย เพราะ State ของ Counter เปลี่ยน แต่ทำไม App.jsx มันถึงถูก render ด้วย ทั้งๆที่ไม่ได้ใช้ state เลย

Zustand console Debug

สำหรับใครที่สังเกตว่ามันมี log 2 รอบ (มันจะมีผลแค่ใน Development Mode เท่านั้น) สามารถเอา StrictMode ออกได้นะครับ

main.jsx
1
ReactDOM.createRoot(document.getElementById('root')).render(
2
// <React.StrictMode>
3
<App />
4
// </React.StrictMode>
5
)

คำตอบคือ ตรงนี้ครับ ที่มีปัญหา

1
const { increment, decrement } = useCounterStore()

อย่างที่ Docs เขียนไว้ครับ ว่าเราไม่ควรใช้แบบนี้ เพราะมันจะมีผลเรื่อง Performance แน่ๆ มันจะ re-render ทุกๆ state ใน store

1
const state = useStore()

วิธีการ ก็คือทำการ select state ออกมาแบบนี้

1
const increment = useCounterStore((state) => state.increment)
2
const decrement = useCounterStore((state) => state.decrement)

หรือถ้าเราต้องการ select หลายๆ state แบบ destructuring object เราสามารถใช้ shallow มาช่วยได้ แบบนี้

1
import shallow from 'zustand/shallow'
2
3
const { count, increment, decrement } = useCounterStore(
4
(state) => ({
5
count: state.count,
6
increment: state.increment,
7
decrement: state.decrement
8
}),
9
shallow
10
)

กลับไปแก้ ไฟล์ Counter.jsx ให้ select state ให้ถูกต้อง ก็จะได้เป็นแบบนี้

components/Counter.jsx
1
import React from 'react'
2
import shallow from 'zustand/shallow'
3
import { useCounterStore } from '../store'
4
5
const Counter = () => {
6
const { count, increment, decrement } = useCounterStore(
7
(state) => ({
8
count: state.count,
9
increment: state.increment,
10
decrement: state.decrement
11
}),
12
shallow
13
)
14
15
console.log(`Counter:render`)
16
return (
17
<section>
18
<h3>Counter</h3>
19
<p>count : {count}</p>
20
<button onClick={increment}>เพิ่ม</button>
21
<button onClick={decrement}>ลบ</button>
22
</section>
23
)
24
}
25
26
export default Counter

และไฟล์ App.jsx ตอนนี้ เป็นแบบนี้

App.jsx
1
import logo from './logo.svg'
2
import './App.css'
3
import { useCounterStore } from './store'
4
import Counter from './components/Counter'
5
6
function App() {
7
const increment = useCounterStore((state) => state.increment)
8
const decrement = useCounterStore((state) => state.decrement)
9
10
console.log(`App:render`)
11
12
return (
13
<div className="App">
14
<header className="App-header">
15
<img src={logo} className="App-logo" alt="logo" />
16
<p>Hello Vite + React!</p>
17
<button type="button" onClick={increment}>
18
Increment
19
</button>
20
<button type="button" onClick={decrement}>
21
Decrement
22
</button>
23
24
<Counter />
25
</header>
26
</div>
27
)
28
}
29
30
export default App

Persist State

นอกจากนี้ เรายังสามารถเก็บ Persist State เพื่อเก็บค่าให้มันอยู่ถาวรใน localStorage หรือ sessionStorage เพราะว่าปกติแล้ว state ใน React มันจะถูก reset ถ้าเรา refersh หน้าเว็บ แต่ถ้าใช้ Persist state มันก็จะไม่ถูก reset เพราะมันอ่านจาก localStorage ที่เราเก็บไว้

วิธีการก็คือใช้ middleware ที่ชื่อ persist ครับ ตัว syntax มันก็ประมาณนี้

1
import create from 'zustand'
2
import { persist } from 'zustand/middleware'
3
4
const store = () => ({ myState: 0 })
5
6
const useStore = create(persist(store))

ตอนนี้ คือผมจะเพิ่ม color และ backgroundColor เพื่อแสดงผลหน้าเว็บ ให้มันมาอ่าน persist state แบบนี้

store.js
1
import { persist } from 'zustand/middleware'
2
3
const themeStore = persist(
4
(set) => ({
5
color: '#222',
6
backgroundColor: '#ff0000',
7
setColor: (color) => set(() => ({ color })),
8
setBackgroundColor: (color) => set(() => ({ backgroundColor: color }))
9
}),
10
{ name: 'my-theme' }
11
)
12
13
export const useThemeStore = create(themeStore)

ต่อมาสร้าง Component ThemeControl เพื่อเอาไว้ set ค่า color นั่นเอง

ThemeControl.jsx
1
import React from 'react'
2
import { useThemeStore } from '../store'
3
4
const ThemeControl = () => {
5
const setColor = useThemeStore((state) => state.setColor)
6
const setBgColor = useThemeStore((state) => state.setBackgroundColor)
7
8
return (
9
<div>
10
<div>
11
<p>
12
setColor <input type="text" onChange={(e) => setColor(e.target.value)} />
13
</p>
14
<p>
15
setBgColor <input type="text" onChange={(e) => setBgColor(e.target.value)} />
16
</p>
17
</div>
18
</div>
19
)
20
}
21
22
export default ThemeControl

สุดท้าย ก็ import <ThemeControl /> ไปใน App.jsx จากนั้นก็เพิ่ม condition ให้มัน render style จาก state

App.jsx
1
import logo from './logo.svg'
2
import './App.css'
3
import { useCounterStore, useThemeStore } from './store'
4
import Counter from './components/Counter'
5
import ThemeControl from './components/ThemeControl'
6
7
function App() {
8
const increment = useCounterStore((state) => state.increment)
9
const decrement = useCounterStore((state) => state.decrement)
10
11
const color = useThemeStore((state) => state.color)
12
const bgColor = useThemeStore((state) => state.backgroundColor)
13
14
console.log(`App:render`)
15
16
return (
17
<div className="App">
18
<header
19
className="App-header"
20
style={{
21
backgroundColor: bgColor,
22
color
23
}}
24
>
25
<img src={logo} className="App-logo" alt="logo" />
26
<p>Hello Vite + React!</p>
27
<button type="button" onClick={increment}>
28
Increment
29
</button>
30
<button type="button" onClick={decrement}>
31
Decrement
32
</button>
33
34
<Counter />
35
36
<ThemeControl />
37
</header>
38
</div>
39
)
40
}
41
42
export default App

Zustand console Debug

ทีนี้เมื่อเราดูที่หน้าเว็บเรา และลอง setColor และ BackgroundColor จะเห็นว่า หน้าเว็บมีการเปลี่ยนสีตาม state และค่า ก็ถูกเก็บลง localStorage นั่นเอง ทำให้ refresh ค่า state ไม่ reset

สรุป

ก็หวังว่าบทความนี้จะมีประโยชน์สำหรับคนที่สนใจ Zustand และการจัดการ State บน React.js แม้ว่า ตัวอย่างส่วนใหญ่ จริงๆ ก็คืออ่านจาก Docs นั่นแหละ และก็ลองนำมาประยุกต์ ปรับใช้งานดูครับ นอกจากนี้ ก็ลองอ่า Recipes เพิ่มเติม รวมถึง TypesScript Guide

Happy Coding ❤️

Authors
avatar

Chai Phonbopit

เป็น Web Dev ในบริษัทแห่งหนึ่ง ทำงานมา 10 ปีกว่าๆ ด้วยภาษาและเทคโนโลยี เช่น JavaScript, Node.js, React, Vue และปัจจุบันกำลังสนใจในเรื่องของ Blockchain และ Crypto กำลังหัดเรียนภาษา Rust

Related Posts