Devahoy Logo
PublishedAt

React

React Hooks คืออะไร? + มาลองหัดใช้กันดีกว่า

React Hooks คืออะไร? + มาลองหัดใช้กันดีกว่า

วันนี้จะมาพูดถึงเรื่อง Hooks กันเนาะ เนื่องจากได้ลองอ่าน Introducing Hooks — React เหตุเพราะไปเห็น Repo ตัวนึง ใน Github ว่าทำไมมันขึ้นมาเป็น Trending ชื่อ react-use มีอะไรน่าสนใจนะ ซึ่งก็อ่านผ่านๆและไม่ได้สนใจอะไรมาก จนกระทั่งมาอ๋อๆๆ ตอนที่เห็น Blog เรื่อง React Hooks นี่แหละ พบว่ามันมีจุดน่าสนใจเป็นอย่างยิ่ง จึงลองอ่าน ทำความเข้าใจ และก็มาบล็อกบอกเล่าเรื่องราวในการทดลองเล่น React Hook แถมเป็นการทบทวนตัวเองไปในตัวด้วย ใครผ่านมาอ่านก็มาช่วยคอมเม้น เสนอแนะได้เนาะ

Hook คืออะไร?

Hook คือ feature ใหม่ที่เพิ่งมีใน React v16.7.0-alpha ซึ่ง Dan Abramov ได้พูดถึงไว้ใน Video นี้ครับ

Link Youtube : https://www.youtube.com/watch?v=V-QO-KO90iQ

จริงๆแล้ว Hook มันก็คือ function ที่ให้เราสามารถใช้ react features ได้ เช่น ให้เราสามารถเรียกใช้ state ได้ โดยที่ไม่ต้องใช้การ implement Class แล้ว

หากใครที่เขียน React มา แล้วใช้แบบ Stateless Component อยู่ๆวันดีคืนดี นึกได้ว่า Component นี้มันต้องใช้ lifecycle ใช้ state ด้วยวุ้ย ก็ต้องนั่งเปลี่ยนจาก function component มาเป็น Class และ extends Component แต่พอเป็น Hook เราไม่ต้องมาเปลี่ยนคลาสแล้ว ก็ใช้ function component ได้เลย

ซึ่งหลายๆข้อทำให้เกิด React Hook ขึ้นมา เช่น

  • Component มีขนาดใหญ่ ยากต่อการ refactor และ test
  • ต้องทำ Logic หรือ Lifecycle ซ้ำๆ ระหว่างแต่ Component
  • Complex Pattern พวก render props หรือ HOC

ซึ่งก็เลยทำให้ Hook มันเกิดมาเพื่อแก้ปัญหา เพื่อ reuse ได้นั่นเอง อ้างอิงจาก Blog ของ Dan Abramov นี้ครับ

กฎการใช้ Hooks

  1. ต้องเรียก Hook ที่ส่วน Top Level ของ function เท่านั้น
  2. ต้องเรียก Hook ภายใน React function เท่านั้น (function components)

Class Component แบบปกติ

ก่อนไปเริ่ม Hooks มาย้อนทำ Class Component แบบพื้นฐาน ในการ handle counter กัน ก็จะหน้าตาประมาณนี้ ตัวอย่าง Class Component ปกติ ที่เวลา เราต้องการให้ Component มี State หรือบางที ก็อาจจะมีการ fetch ข้อมูลด้วย componentDidMount() ก็จะเป็นลักษณะแบบนี้

1
import React, { Component } from 'react'
2
class ExampleHook extends Component {
3
constructor(props) {
4
super(props)
5
this.state = {
6
count: 0
7
}
8
}
9
10
componentDidMount() {
11
// fetch some data
12
fetchData()
13
}
14
15
handleClick = () => {
16
this.setState((state) => ({
17
count: state.count + 1
18
}))
19
}
20
21
render() {
22
const { count } = this.state
23
return (
24
<div>
25
<p>You clicked {count} times</p>
26
<button onClick={this.handleClick}>Click me</button>
27
</div>
28
)
29
}
30
}

ซึ่งวิธีทั่วๆไป ก็คือเราทำการสร้าง Class จากนั้น ก็กำหนด this.state = {} ขึ้นมา พร้อมทั้ง ทำ handle function เพื่อเอาไว้ handle onclick

เรียนรู้ Hooks

มาลองเรียนรู้หน้าตาเจ้า Hooks กันดีกว่า ว่ามันจะมีหน้าตาเป็นยังไง และวิธีการเรียกใช้แบบไหนบ้าง

useState

ตัวอย่างแรกของ Hooks ก็คือ State Hook ที่ชื่อว่า useState

1
import React, { useState } from 'react'
2
function ExampleHook() {
3
const [count, setCount] = useState(0)
4
5
render() {
6
return (
7
<div>
8
<p>You clicked {count} times</p>
9
<button onClick={() => setCount(count + 1)}>
10
Click me
11
</button>
12
</div>
13
);
14
}
15
}
  • useState คือ Hook โดยเราเรียกมันภายใน react function component
  • useState return ค่าเป็น array ตัวแรกคือ ชื่อ state และ ตัวสองคือชื่อ function
  • เราใช้ array destructuring เพื่อกำหนด ชื่อ state และ function ที่ return จาก useState

กรณีที่เราต้องการใช้ หลายๆ state ก็ทำได้ เช่น

1
function ExampleWithManyStates() {
2
// Declare multiple state variables!
3
const [age, setAge] = useState(42)
4
const [fruit, setFruit] = useState('banana')
5
const [todos, setTodos] = useState([{ text: 'Learn Hooks' }])
6
// ...
7
}

useEffect

ต่อมา useEffect เป็น Hook ที่มี side effect ก็คือมีปัจจัยภายนอกเข้ามาเกี่ยวข้องกับตัว function component เช่น การแก้ไขพวก window, title หรือเรียก lifecycle ของ React เช่น componentDidMount

ตัวอย่างการใช้ useEffect

1
import { useState, useEffect } from 'react'
2
3
function Example() {
4
const [count, setCount] = useState(0)
5
6
useEffect(() => {
7
document.title = `You clicked ${count} times`
8
})
9
10
return (
11
<div>
12
<p>You clicked {count} times</p>
13
<button onClick={() => setCount(count + 1)}>Click me</button>
14
</div>
15
)
16
}
  • useEffect() : เป็น function ที่รับ function เป็น argument ตัวอย่างด้านบน คือ รับ function ที่ทำการเปลี่ยนค่า document.title
  • useEffect() : ถูกรันทุกๆครั้งที่มีการ render รวมถึงการ render ครั้งแรก (ทุกครั้งที่มีการอัพเดทนั่นเอง)

มาลองเล่น Hooks กันดีกว่า

เริ่มแรก จำเป็นต้อง ติดตั้ง React เวอร์ชั่น v16.7.0-alpha ขึ้นไปครับ ถ้าเราติดตั้งแบบ yarn add react มันจะเอาตัว stable ที่ไม่ใช่ alpha ครับ

มาเริ่มต้นด้วยการสร้างโปรเจ็คด้วย Create React App และติดตั้ง React เวอร์ชั่นล่าสุดกันครับ

Terminal window
npx create-react-app hello-hooks
cd hello-hooks

จากนั้นที่ไฟล์ package.json ให้แก้เวอร์ชัน React เป็น ^16.7.0-alpha.2

1
"dependencies": {
2
"react": "^16.7.0-alpha.2",
3
"react-dom": "^16.7.0-alpha.2",
4
"react-scripts": "2.1.1"
5
}

Install dependencies ใหม่ซะ

1
yarn install

จากนั้นเปิดไฟล์ src/App.js และใช้ useState() แบบนี้

1
import React, { useState } from 'react'
2
import logo from './logo.svg'
3
import './App.css'
4
5
const App = () => {
6
const [count, setCount] = useState(0)
7
8
return (
9
<div className="App">
10
<header className="App-header">
11
<img src={logo} className="App-logo" alt="logo" />
12
<p>
13
Edit <code>src/App.js</code> and save to reload.
14
</p>
15
16
<p>You click {count} times</p>
17
<button
18
style={{
19
padding: '8px 16px',
20
borderRadius: 4,
21
fontSize: '1.25rem'
22
}}
23
onClick={() => setCount(count + 1)}
24
>
25
Click me
26
</button>
27
</header>
28
</div>
29
)
30
}
31
32
export default App

เราจะได้หน้าจอ และมีปุ่ม Button สำหรับ setState ด้วยการใช้ useState() นั่นเอง ทีนี้เราก็สามารถใช้ Hooks ได้แล้ว

useState

ต่อมาลองทำ useEffect() เพื่อ fetch ข้อมูลจาก Random Cat API แทน การใช้ componentDidMount() กันดูบ้าง

1
const [cat, setCat] = useState({})
2
3
const randomCat = () => axios.get('https://aws.random.cat/meow')
4
5
useEffect(() => {
6
randomCat().then((response) => {
7
setCat(response.data)
8
})
9
})
10
11
// ใส่ log เพื่อดูว่ามัน render ยังไง
12
console.log('render >>>')
13
14
return (
15
<div className="App">
16
<header className="App-header">
17
<img src={logo} className="App-logo" alt="logo" />
18
<p>
19
Edit <code>src/App.js</code> and save to reload.
20
</p>
21
22
<p>You click {state.count} times</p>
23
24
<button
25
style={{
26
padding: '8px 16px',
27
borderRadius: 4,
28
fontSize: '1.25rem'
29
}}
30
onClick={() => {
31
dispatch({
32
type: 'COUNTER_CLICK',
33
payload: state.count + 1
34
})
35
}}
36
>
37
Click me
38
</button>
39
40
<p>
41
<img src={state.cat.file} alt="Cat" width="256" />
42
</p>
43
</header>
44
</div>
45
)

ทดลองดูหน้าเว็บ แต่ๆ

useEffect() loop

เว็บเรามันทำการ render รัวๆเลย ก็เพราะว่าโดยปกติ useEffect() มันถูกเรียกทุกๆครั้งที่ทำการ render ถ้าเราจะไม่ให้มัน รันใหม่ ให้เฉพาะตอน mount และ unmont เราก็ต้องส่ง argument ตัวที่สองเป็น array ไปให้มันด้วย เช่น ส่ง empty array ไปแบบนี้

หรือจะใส่ ค่า state ที่เราไว้เช็คว่า ค่าใน array มีการเปลียนแปลง มันก็จะเรียก useEffect อีกครั้ง เช่น useEffect(fn, [someValue])

1
const fn = () => {
2
randomCat().then()
3
}
4
5
useEffect(fn, [])

สุดท้ายไฟล์ src/App.js จะได้แบบนี้

1
import React, { useState, useEffect } from 'react'
2
import axios from 'axios'
3
import logo from './logo.svg'
4
import './App.css'
5
6
const randomCat = () => axios.get('https://aws.random.cat/meow')
7
8
const App = () => {
9
const [count, setCount] = useState(0)
10
const [cat, setCat] = useState({})
11
12
useEffect(() => {
13
randomCat().then((response) => {
14
setCat(response.data)
15
})
16
}, [])
17
18
return (
19
<div className="App">
20
<header className="App-header">
21
<img src={logo} className="App-logo" alt="logo" />
22
<p>
23
Edit <code>src/App.js</code> and save to reload.
24
</p>
25
26
<p>You click {count} times</p>
27
28
<button
29
style={{
30
padding: '8px 16px',
31
borderRadius: 4,
32
fontSize: '1.25rem'
33
}}
34
onClick={() => setCount(count + 1)}
35
>
36
Click me
37
</button>
38
39
<p>
40
<img src={cat.file} alt="Meow" width="256" />
41
</p>
42
</header>
43
</div>
44
)
45
}
46
47
export default App

useReducer

แถมครับ นอกจากนี้ ตัว Hooks ยังมี useReducer มาให้เรา สำหรับคนที่เขียน Redux มา ก็คงคุ้นเคยดีอยู่แล้ว นั่นคือ เราสามารถใช้ Redux ใน Hooks ได้เลย ตัว Reducers ก็ไม่แตกต่างจาก Redux ครับ

หากใครไม่เคยเขียน Redux แนะนำอ่านบทความนี้เพิ่มเติมเนอะ Redux คืออะไร? + เริ่มต้นเรียนรู้ Redux ร่วมกับ React กันดีกว่า

วิธีใช้งานก็ประมาณนี้ สร้าง reducers ของเราขึ้นมา

1
const initialState = {
2
isFetching: false,
3
cat: {}
4
}
5
6
const reducer = (state, { type, payload }) => {
7
switch (type) {
8
case 'FETCH_CAT_PENDING':
9
return {
10
...state,
11
isFetching: true
12
}
13
case 'FETCH_CAT_SUCCESS':
14
return {
15
...state,
16
isFetching: false,
17
cat: payload
18
}
19
default:
20
return state
21
}
22
}
23
24
// การใช้การ ก็คือเรียก useReducer ด้วย function reducer และ initiState เป็น argument ครับ
25
26
const [state, dispatch] = useReducer(reducer, initialState)
  • useReducer : รับ reducer และ initiState เป็น argument
  • useReducer : return ค่ากลับมาเป็น state และ dispatch ครับ

โดยเราลองแก้ไขตัว src/App.js ให้ใช้ state ด้วย useReducer แทน useState ดีกว่า

1
import React, { useReducer, useEffect } from 'react'
2
import axios from 'axios'
3
import logo from './logo.svg'
4
import './App.css'
5
6
const fetchCat = () => axios.get('https://aws.random.cat/meow')
7
8
const initialState = {
9
isFetching: false,
10
cat: {},
11
count: 0
12
}
13
14
const reducer = (state, { type, payload }) => {
15
switch (type) {
16
case 'FETCH_CAT_PENDING':
17
return {
18
...state,
19
isFetching: true
20
}
21
case 'FETCH_CAT_SUCCESS':
22
return {
23
...state,
24
isFetching: false,
25
cat: payload
26
}
27
case 'COUNTER_CLICK':
28
return {
29
...state,
30
isFetching: false,
31
count: payload
32
}
33
default:
34
return state
35
}
36
}
37
38
const App = () => {
39
const [state, dispatch] = useReducer(reducer, initialState)
40
41
useEffect(() => {
42
dispatch({
43
type: 'FETCH_CAT_PENDING'
44
})
45
46
fetchCat().then((response) => {
47
dispatch({
48
type: 'FETCH_CAT_SUCCESS',
49
payload: response.data
50
})
51
})
52
}, [])
53
54
if (state.isFetching) {
55
return <p>Loading....</p>
56
}
57
58
return (
59
<div className="App">
60
<header className="App-header">
61
<img src={logo} className="App-logo" alt="logo" />
62
<p>
63
Edit <code>src/App.js</code> and save to reload.
64
</p>
65
66
<p>You click {state.count} times</p>
67
68
<button
69
style={{
70
padding: '8px 16px',
71
borderRadius: 4,
72
fontSize: '1.25rem'
73
}}
74
onClick={() => {
75
dispatch({
76
type: 'COUNTER_CLICK',
77
payload: state.count + 1
78
})
79
}}
80
>
81
Click me
82
</button>
83
84
<p>
85
<img src={state.cat.file} alt="Cat" width="256" />
86
</p>
87
</header>
88
</div>
89
)
90
}
91
92
export default App

และเนื่องจาก เราสามารถใช้ Object destructuring ได้ ก็ ปรับตรง useReducer เป็นแบบนี้ซะ จะได้ไม่ต้องเรียก state.xxx บ่อยๆ

1
const [{ cat, isFetching, count }, dispatch] = useReducer(reducer, initialState)

สุดท้ายไฟล์​ src/App.js จะได้แบบนี้

1
import React, { useReducer, useEffect } from 'react'
2
import axios from 'axios'
3
import logo from './logo.svg'
4
import './App.css'
5
6
const fetchCat = () => axios.get('https://aws.random.cat/meow')
7
8
const initialState = {
9
isFetching: false,
10
cat: {},
11
count: 0
12
}
13
14
const reducer = (state, { type, payload }) => {
15
switch (type) {
16
case 'FETCH_CAT_PENDING':
17
return {
18
...state,
19
isFetching: true
20
}
21
case 'FETCH_CAT_SUCCESS':
22
return {
23
...state,
24
isFetching: false,
25
cat: payload
26
}
27
case 'COUNTER_CLICK':
28
return {
29
...state,
30
isFetching: false,
31
count: payload
32
}
33
default:
34
return state
35
}
36
}
37
38
const App = () => {
39
const [{ cat, isFetching, count }, dispatch] = useReducer(reducer, initialState)
40
41
useEffect(() => {
42
dispatch({
43
type: 'FETCH_CAT_PENDING'
44
})
45
46
fetchCat().then((response) => {
47
dispatch({
48
type: 'FETCH_CAT_SUCCESS',
49
payload: response.data
50
})
51
})
52
}, [])
53
54
if (isFetching) {
55
return <p>Loading....</p>
56
}
57
58
return (
59
<div className="App">
60
<header className="App-header">
61
<img src={logo} className="App-logo" alt="logo" />
62
<p>
63
Edit <code>src/App.js</code> and save to reload.
64
</p>
65
66
<p>You click {count} times</p>
67
68
<button
69
style={{
70
padding: '8px 16px',
71
borderRadius: 4,
72
fontSize: '1.25rem'
73
}}
74
onClick={() => {
75
dispatch({
76
type: 'COUNTER_CLICK',
77
payload: count + 1
78
})
79
}}
80
>
81
Click me
82
</button>
83
84
<p>
85
<img src={cat && cat.file} alt="Cat" width="256" />
86
</p>
87
</header>
88
</div>
89
)
90
}
91
92
export default App

ทดลองรันหน้าเว็บอีกครั้ง

Finish

สุดท้าย

นอกเหนือจากตัวอย่างที่กล่าวมา ตัว React Hooks ก็ยังมีอะไรให้ทดลองเล่นอีกเยอะครับ และเนื่องจากว่ามันก็ยังเป็นตัว Beta ฉะนั้น เมื่อตอนที่ release แล้ว syntax หรือวิธีการเขียน ก็อาจจะแตกต่างจากตอนปัจจุบันก็ได้ ฉะนั้นใครที่ลองเล่นกับมันอยู่ ก็ต้องเผื่อจุดนี้ไว้ด้วยนะครับ

ตัวอย่าง Hooks ต่างๆ ที่น่าสนใจ รวมเป็น Collection หรือ Guide เช่น

Recompose

และๆ ตัว Recompose ที่หลายๆคนชอบใช้กัน เค้าเลิก maintenance แล้ว จะไม่มี feature ใหม่ๆแต่ยังใช้ได้อยู่เนาะ เพราะตัวคนที่เค้าเริ่มทำ Recompose ขึ้นมาเพื่อแก้ปัญหาต่างๆ ตัว React Hooks มันตอบโจทย์หมดแล้ว ก็เลยคิดว่า น่าจะเริ่มไปใช้ Hooks กันแน่ๆครับ ก็แน่ละ คนทำ recompose เค้าก็ทำงานอยู่ Facebook นี่เนาะ

Happy Coding ❤️
Authors
avatar

Chai Phonbopit

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

Related Posts