* 리액트 훅의 개념
리액트 버전 16.8 부터 새로 추가된 기능이다. 기존의 함수형 컴포넌트는 props 만 전달받아 렌더링할 뿐 state 를 사용하여 상태값을 변경할 수 없었다. 또한, 라이프사이클 메서드도 사용하지 못하였다. 함수형 컴포넌트에서도 이를 가능하게 하는 것이 바로 리액트 훅이다.
* 리액트 훅을 사용하는 이유
상태관련 로직을 독립적으로 분리하기 위함이다. 마치 관련된 코드를 함수로 묶어서 관리하는 것과 유사하다. 즉, 서로 관련된 상태로직을 분리하고 모듈화해서 유지보수하기 좋게 한다. 또한, 클래스형 컴포넌트에서 자주 사용하는 자바스크립트 this 키워드는 상황에 따라 다른 값을 가지게 되므로 개발자에게 혼란을 초래한다. 함수형 컴포넌트에서는 this 를 사용할 필요가 없다.
* 클래스형 컴포넌트와 함수형 컴포넌트의 state 관리 비교 - State Hook
버튼을 클릭할때 숫자를 증가시키는 카운터 예제를 다시 살펴보자!
import './App.css';
import React, { Component } from 'react';
import Button from './Button'
class App extends Component {
state = {
count: 0
}
increaseCount = () => {
this.setState({ count: this.state.count + 1 })
}
render(){
const { count } = this.state
return (
<div className="App">
<h1>Count : {count}</h1>
<Button handleClick={this.increaseCount}>Increase Count number</Button>
</div>
);
}
}
export default App;
App.js 파일을 위와 같이 작성하자!
state = {
count: 0
}
카운팅하는 숫자를 변경하기 위하여 count 상태를 사용한다.
increaseCount = () => {
this.setState({ count: this.state.count + 1 })
}
버튼을 클릭할때 실행되는 이벤트핸들러 함수이다. 버튼을 클릭할때마다 count 상태를 1씩 증가시킨다.
render(){
const { count } = this.state
return (
<div className="App">
<h1>Count : {count}</h1>
<Button handleClick={this.increaseCount}>Increase Count number</Button>
</div>
);
}
count 상태를 조회한 다음 화면에 렌더링한다. 버튼을 클릭하면 click 이벤트가 발생하면서 increaseCount 함수가 실행된다.
그럼 같은 기능을 함수형 컴포넌트에서 구현하려면 어떻게 하면 될까?
import './App.css';
import React, { useState } from 'react';
import Button from './Button'
function App () {
const [count, setCount] = useState(0)
console.log("카운트: ", count)
const increaseCount = () => {
setCount(count + 1)
}
return (
<div className="App">
<h1>Count : {count}</h1>
<Button handleClick={increaseCount}>Increase Count number</Button>
</div>
);
}
export default App;
App.js 파일을 위와 같이 수정하자! App 컴포넌트를 함수형으로 전환한다.
import React, { useState } from 'react';
react 라이브러리에서 useState 함수를 임포트한다. 해당 함수는 함수형 컴포넌트에서 상태(state) 관리를 할 수 있도록 해준다.
const [count, setCount] = useState(0)
useState 함수의 인자로 0 을 전달하여 count 상태를 초기화한다. 해당 함수는 count 상태와 count 상태를 변경하는 setCount 함수를 반환한다. setCount 함수는 클래스형 컴포넌트의 setState 메서드와 동일한 역할을 한다.
const increaseCount = () => {
setCount(count + 1)
}
버튼이 클릭될때마다 setCount 함수를 사용하여 count 상태를 1씩 증가시킨다.
<div className="App">
<h1>Count : {count}</h1>
<Button handleClick={increaseCount}>Increase Count number</Button>
</div>
count 값을 조회하고 화면에 렌더링한다. 더이상 상태 (state) 를 this.state 로 접근하지 않아도 된다.
한가지 주목할점은 리렌더링할때 아래와 같이 함수로 정의한 컴포넌트 전체가 새로 실행된다는 점이다.
사용자 정보와 할일목록을 화면에 보여주는 다른 예제를 살펴보자!
import './App.css';
import React, { Component } from 'react';
import Button from './Button'
class App extends Component {
state = {
user: {
name: 'syleemomo',
age: 23,
fruits: ["apple", "banana", "orange"]
},
todos: [
{title: 'cleaning', done: false, description: 'cleaning my living room'},
{title: 'learning', done: false, description: 'learing react on tomorrow morning'},
{title: 'drinking', done: false, description: 'drinking soju with close friends'}
]
}
changeName = () => {
const newUser = {...this.state.user, name: "new name"}
this.setState({user: newUser})
}
addNewTodo = () => {
const newTodo = {
title: 'checking', done: true, description: 'checking my state of score'
}
const todos = [...this.state.todos, newTodo]
this.setState({todos})
}
render(){
const { user, todos } = this.state
const { changeName, addNewTodo } = this
return (
<div className="App">
<h1>User Information</h1>
<h2>{user.name} ({user.age})</h2>
<h3>favorite food: {user.fruits.join(" ")}</h3>
<Button handleClick={changeName}>Change Name</Button>
<h1>Todo List</h1>
{todos.map( (todo, id) => {
return (
<div key={id}>
<h2>{todo.title} - ({todo.done? "finished": "not yet done"})</h2>
<p>{todo.description}</p>
</div>
)
})}
<Button handleClick={addNewTodo}>Add Todo</Button>
</div>
);
}
}
export default App;
App.js 파일을 위와 같이 수정하자!
state = {
user: {
name: 'syleemomo',
age: 23,
fruits: ["apple", "banana", "orange"]
},
todos: [
{title: 'cleaning', done: false, description: 'cleaning my living room'},
{title: 'learning', done: false, description: 'learing react on tomorrow morning'},
{title: 'drinking', done: false, description: 'drinking soju with close friends'}
]
}
사용자 정보에 대한 상태와 할일목록에 대한 상태가 state 라는 하나의 변수 안에 섞여있다.
changeName = () => {
const newUser = {...this.state.user, name: "new name"}
this.setState({user: newUser})
}
사용자의 이름을 변경하는 이벤트핸들러 함수이다. 스프레드 연산자(...)를 사용하여 기존의 user 정보를 새로운 객체에 풀어헤친 다음 name 프로퍼티의 값만 new name 으로 변경한다.
addNewTodo = () => {
const newTodo = {
title: 'checking', done: true, description: 'checking my state of score'
}
const todos = [...this.state.todos, newTodo]
this.setState({todos})
}
새로운 할일을 추가하는 이벤트핸들러 함수이다. newTodo 라는 새로운 할일을 생성한다. 스프레드 연산자(...)를 사용하여 기존의 todos 를 새로운 배열에 풀어헤친 다음 새로운 배열요소인 newTodo 를 추가한다. setState 메서드로 todos 상태를 업데이트한다.
render(){
const { user, todos } = this.state
const { changeName, addNewTodo } = this
return (
<div className="App">
<h1>User Information</h1>
<h2>{user.name} ({user.age})</h2>
<h3>favorite food: {user.fruits.join(" ")}</h3>
<Button handleClick={changeName}>Change Name</Button>
<h1>Todo List</h1>
{todos.map( (todo, id) => {
return (
<div key={id}>
<h2>{todo.title} - ({todo.done? "finished": "not yet done"})</h2>
<p>{todo.description}</p>
</div>
)
})}
<Button handleClick={addNewTodo}>Add Todo</Button>
</div>
);
}
사용자 정보와 할일목록에 대한 상태(state) 를 조회한 다음 화면에 해당 값들을 렌더링한다.
그럼 위 코드와 동일하게 동작하면서 함수형 컴포넌트로 전환하려면 어떻게 하면 될까?
import './App.css';
import React, { useState } from 'react';
import Button from './Button'
function App() {
const [user, setUser] = useState({
name: 'syleemomo',
age: 23,
fruits: ["apple", "banana", "orange"]
})
const [todos, setTodos] = useState([
{title: 'cleaning', done: false, description: 'cleaning my living room'},
{title: 'learning', done: false, description: 'learing react on tomorrow morning'},
{title: 'drinking', done: false, description: 'drinking soju with close friends'}
])
const changeName = () => {
const newUser = {...user, name: "new name"}
setUser(newUser)
}
const addNewTodo = () => {
const newTodo = {
title: 'checking', done: true, description: 'checking my state of score'
}
const newTodos = [...todos, newTodo]
setTodos(newTodos)
}
return (
<div className="App">
<h1>User Information</h1>
<h2>{user.name} ({user.age})</h2>
<h3>favorite food: {user.fruits.join(" ")}</h3>
<Button handleClick={changeName}>Change Name</Button>
<h1>Todo List</h1>
{todos.map( (todo, id) => {
return (
<div key={id}>
<h2>{todo.title} - ({todo.done? "finished": "not yet done"})</h2>
<p>{todo.description}</p>
</div>
)
})}
<Button handleClick={addNewTodo}>Add Todo</Button>
</div>
);
}
export default App;
App.js 파일을 위와 같이 수정하자!
import React, { useState } from 'react';
react 라이브러리에서 useState 함수를 임포트한다.
// 사용자 정보에 대한 상태를 관리하는 로직
const [user, setUser] = useState({
name: 'syleemomo',
age: 23,
fruits: ["apple", "banana", "orange"]
})
// 할일 목록에 대한 상태를 관리하는 로직
const [todos, setTodos] = useState([
{title: 'cleaning', done: false, description: 'cleaning my living room'},
{title: 'learning', done: false, description: 'learing react on tomorrow morning'},
{title: 'drinking', done: false, description: 'drinking soju with close friends'}
])
useState 함수를 이용하여 사용자 정보와 할일목록에 대한 상태를 초기화한다. 클래스형 컴포넌트에서는 사용자 정보에 대한 상태와 할일목록에 대한 상태가 state 라는 하나의 변수 안에 섞여 있었지만 리액트 훅은 이를 분리한다. 즉, 관련된 상태와 해당 상태를 변경하는 로직을 하나로 묶고, 독립적으로 분리하여 유지보수와 코드 가독성에 도움을 준다.
const changeName = () => {
const newUser = {...user, name: "new name"}
setUser(newUser)
}
사용자 이름을 변경하는 이벤트핸들러 함수이다. 스프레드 연산자(...)를 사용하여 기존의 user 상태를 새로운 객체에 풀어헤친 다음 name 프로퍼티만 new name 을 변경한다. 클래스형 컴포넌트처럼 this.state 로 접근할 필요가 없다. setUser 함수를 이용하여 user 상태를 업데이트한다.
const addNewTodo = () => {
const newTodo = {
title: 'checking', done: true, description: 'checking my state of score'
}
const newTodos = [...todos, newTodo]
setTodos(newTodos)
}
새로운 할일을 추가하는 이벤트핸들러 함수이다. 스프레드 연산자(...)를 사용하여 기존의 todos 상태를 새로운 배열에 풀어헤친 다음 newTodo 객체를 새로운 배열요소로 추가한다. 클래스형 컴포넌트처럼 this.state 로 접근할 필요가 없다. setTodos 함수를 이용하여 todos 상태를 업데이트한다.
return (
<div className="App">
<h1>User Information</h1>
<h2>{user.name} ({user.age})</h2>
<h3>favorite food: {user.fruits.join(" ")}</h3>
<Button handleClick={changeName}>Change Name</Button>
<h1>Todo List</h1>
{todos.map( (todo, id) => {
return (
<div key={id}>
<h2>{todo.title} - ({todo.done? "finished": "not yet done"})</h2>
<p>{todo.description}</p>
</div>
)
})}
<Button handleClick={addNewTodo}>Add Todo</Button>
</div>
);
user, todos 상태를 조회하고 화면에 해당 값들을 렌더링한다.
* 함수형 컴포넌트에서 라이프사이클 메서드 사용하기 - Effect Hook
Effect Hook 은 우선 컴포넌트가 초기 렌더링될때 비동기 함수로 자바스크립트에 의해 등록해둔다. 이후 상태가 변경되서 리렌더링될때마다 실행되거나 초기에 한번만 실행될수도 있다.
Effect Hook 의 두번째 의존성 배열이 빈 배열이 아닌 경우 Effect Hook 안의 콜백함수는 의존성 배열에 주어진 state 값이 변경될때마다 새로 업데이트된 state 값을 삽입하여 콜백함수를 새로 생성한다.
import './App.css';
import React, { useEffect, useState } from 'react';
import Button from './Button'
function App() {
const [count, setCount] = useState(0);
const increaseCount = () => {
setCount(count + 1)
}
// componentDidMount, componentDidUpdate 와 유사함
useEffect( () => {
document.title = `You clicked ${count} times`
}, [count])
return (
<div className="App">
<h1>Count: {count}</h1>
<Button handleClick={increaseCount}>Increase Number</Button>
</div>
);
}
export default App;
App.js 파일을 위와 같이 작성하자!
import React, { useEffect, useState } from 'react';
react 라이브러리에서 useEffect 함수를 임포트한다. useEffect 함수가 Effect Hook 이다. 컴포넌트가 초기에 렌더링되고 나서 수행해야 할 작업이나 업데이트가 완료된 이후에 수행해야 할 작업을 side effects (effects) 라고 한다. 이러한 작업을 Effect Hook 에 콜백함수 형태로 전달하면 된다.
// componentDidMount, componentDidUpdate 와 유사함
useEffect( () => {
document.title = `You clicked ${count} times`
}, [count])
useEffect 함수에 콜백함수를 전달한다. 콜백함수는 side effects 이며, HTML 문서의 title 을 변경한다. 또한, 콜백함수는 초기 렌더링이 완료된 시점이나 count 상태가 업데이트될 때마다 실행된다. useEffect 함수의 두번째 인자로 주어진 [count] 는 count 상태가 업데이트될때마다 업데이트된 최신 count 값으로 콜백함수를 실행한다. 즉, 클래스형 컴포넌트에서 사용하는 라이프사이클 메서드인componentDidMount 나 componentDidUpdate 의 기능을 통합한 것이다.
import React, { useState, useEffect } from 'react'
import logo from './logo.svg';
import './App.css';
import Button from './Button'
// useEffect(콜백함수, 의존성 배열)
// 의존성 배열: dependency array
// 의존성 배열이 빈 배열인 경우: componentDidMount
// 의존성 배열에 아무것도 설정하지 않은 경우: componentDidUpdate
// 의존성 배열에 state, props 를 설정한 경우: 해당 state, props 가
// 변경될때마다 콜백함수가 실행됨 : componentDidUpdate
function App (){
const [count, setCount] = useState(0)
const [name, setName] = useState('')
const increaseCount = () => {
setCount(count+1)
}
const changeName = () => {
setName(name + 'a')
}
// componentDidMount or componentDidUpdate
useEffect(() => {
console.log("카운트 콜백!")
document.title = `You clicked ${count} times`
}, [name, count]) // count 값이 업데이트될때마다 콜백함수 실행
return (
<div className='App'>
<h1>Count: {count}</h1>
<h1>Name: {name}</h1>
<Button handleClick={increaseCount}>Increase Number</Button>
<Button handleClick={changeName}>이름 변경</Button>
</div>
)
}
export default App
의존성 배열에 아무런 설정을 하지 않으면, 해당 컴포넌트 안의 어떤 state 나 props 가 변경되더라도 리렌더링될때마다 useEffect 안의 콜백함수가 실행이 된다. 하지만, 특정 state 나 props 를 설정하게 되면 해당 state 나 props 가 변경될때만 콜백함수가 실행된다.
컴포넌트 생명주기 예제코드 중 1초마다 자동으로 숫자를 카운팅하는 예제를 Effect Hook 으로 만들어보자!
import './App.css';
import React, { useEffect, useState } from 'react';
function App() {
const [count, setCount] = useState(0);
useEffect( () => {
const increaseCount = () => {
clearTimeout(timerID)
setCount(count + 1)
}
const timerID = setTimeout(increaseCount, 1000)
return () => {
clearTimeout(timerID)
}
}, [count]) // count 값이 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
return (
<div className="App">
<h1>Increase Count automatically !</h1>
<h2>Count: {count}</h2>
</div>
);
}
export default App;
App.js 파일을 위와 같이 작성하자!
const [count, setCount] = useState(0);
useState 함수를 사용하여 count 상태를 0 으로 초기화한다.
const increaseCount = () => {
clearTimeout(timerID)
setCount(count + 1)
}
setCount 함수를 사용하여 count 상태를 1씩 증가시킨다. 해당 함수는 useEffect 외부에 있어도 동작하지만 1초 전에 설정한 타이머를 해제해주기 위하여 useEffect 안에 정의한다. 왜냐하면 timerID 값은 useEffect 외부에서 조회가 불가능하기 때문이다.
useEffect( () => {
const increaseCount = () => {
clearTimeout(timerID)
setCount(count + 1)
}
const timerID = setTimeout(increaseCount, 1000)
return () => {
clearTimeout(timerID)
}
}, [count]) // count 값이 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
초기 렌더링이 끝나면 브라우저 API 인 setTimeout 함수를 사용하여 1초후에 increaseCount 함수를 실행할 타이머를 설정한다. 1초 후에 increaseCount 함수가 실행되면 1초 전에 설정한 타이머를 먼저 해제해주고, count 값을 1만큼 증가시킨다. count 상태가 업데이트되면 리렌더링이 일어나서 새로운 count 값을 화면에 보여주고, useEffect 함수 안에 정의한 콜백함수가 재실행된다. 그러면 다시 타이머가 설정되어서 앞서 수행한 과정이 반복된다. 즉, 1초 후에 increaseCount 함수를 실행하여 카운트값을 1증가시킨다. 결국 1초마다 count 상태가 1씩 증가하게 된다.
useEffect 의 두번째 인자 count 는 의존성 배열(dependency array)이라고 하며, count 상태가 변경될때마다 useEffect 에 설정된 콜백함수를 실행하겠다는 의미이다.
클래스형 컴포넌트에서 컴포넌트가 unmount 될때 실행할 코드는 componentWillUnmount 라이프사이클 메서드 안에 구현하였다. 함수형 컴포넌트에서 컴포넌트가 unmount 될때 실행할 코드는 useEffect 함수에서 콜백함수 형태로 반환해주면 된다. 이를 클린업(Clean-up)이라고 한다.
좀 더 자세히 설명하면 아래와 같다.
1. count 값이 0 으로 초기화된다.
2. useEffect 안의 콜백함수는 아래와 같이 메모리에 생성된다.
useEffect( () => {
const increaseCount = () => {
clearTimeout(timerID)
setCount(0 + 1)
}
const timerID = setTimeout(increaseCount, 1000)
return () => {
clearTimeout(timerID)
}
}, [0]) // count 값이 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
3. 아래와 같이 화면에 count 값을 디스플레이한다.
<div className="App">
<h1>Increase Count automatically !</h1>
<h2>Count: {0}</h2>
</div>
4. 아래의 콜백함수가 실행되면서 타이머를 설정한다.
useEffect( () => {
const increaseCount = () => {
clearTimeout(timerID)
setCount(0 + 1)
}
const timerID = setTimeout(increaseCount, 1000)
return () => {
clearTimeout(timerID)
}
}, [0]) // count 값이 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
5. 타이머는 1초 후에 count 상태를 1만큼 증가시켜서 1이 된다.
6. App 컴포넌트가 재실행되면서 아래와 같이 useEffect 안의 콜백함수가 새로운 count 값을 삽입하여 메모리에 새로 생성된다.
useEffect( () => {
const increaseCount = () => {
clearTimeout(timerID)
setCount(1 + 1)
}
const timerID = setTimeout(increaseCount, 1000)
return () => {
clearTimeout(timerID)
}
}, [1]) // count 값이 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
7. 아래와 같이 화면에 count 값을 디스플레이한다.
<div className="App">
<h1>Increase Count automatically !</h1>
<h2>Count: {1}</h2>
</div>
8. 아래의 콜백함수가 실행되면서 타이머를 재설정한다.
useEffect( () => {
const increaseCount = () => {
clearTimeout(timerID)
setCount(1 + 1)
}
const timerID = setTimeout(increaseCount, 1000)
return () => {
clearTimeout(timerID)
}
}, [1]) // count 값이 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
9. 타이머는 1초 후에 count 상태를 1만큼 증가시켜서 2가 된다.
10. 다시 6번으로 돌아가서 순차적으로 실행된다. 물론 메모리에 생성되는 useEffect 안의 콜백함수에는 새로운 count 값이 삽입된다.
컴포넌트 안에서 여러개의 Effect Hook 을 사용할 수도 있다. 카운팅 예제와 랜덤숫자 예제를 혼합한 다음 코드를 살펴보자!
import './App.css';
import React, { useEffect, useState } from 'react';
function App() {
// 1초마다 자동으로 숫자를 카운팅하는 로직
const [count, setCount] = useState(0);
useEffect( () => {
const increaseCount = () => {
clearTimeout(timerID)
setCount(count + 1)
}
const timerID = setTimeout(increaseCount, 1000)
return () => {
clearTimeout(timerID)
}
}, [count]) // count 상태가 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
// 1초마다 자동으로 랜덤한 숫자를 보여주는 로직
const [number, setNumber] = useState(0)
useEffect( () => {
const pickRandomNumber = () => {
clearTimeout(timerID)
const randNum = Math.floor(Math.random()*100)
setNumber(randNum)
}
const timerID = setTimeout(pickRandomNumber, 1000)
return () => {
clearTimeout(timerID)
}
}, [count]) // count 상태가 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
return (
<div className="App">
<h1>Increase Count automatically !</h1>
<h2>Count: {count}</h2>
<br/>
<h1>Pick Random Number !</h1>
<h2>Random Numbr: {number}</h2>
</div>
);
}
export default App;
App.js 를 위와 같이 작성하자! 여러개의 Effect (리액트 훅) 를 사용하여 숫자 카운팅 로직과 랜덤숫자 로직을 분리하였다. 이것이 리액트 훅의 핵심이다. 예전에는 state 나 라이프사이클 메서드에 관련없는 상태와 관련없는 로직이 뒤섞여 있었다. 리액트 훅은 서로 관련된 state 와 라이프사이클 로직을 묶어서 독립적으로 분리함으로써 유지보수와 코드 가독성을 향상시켜준다.
// 1초마다 자동으로 숫자를 카운팅하는 로직
const [count, setCount] = useState(0);
useEffect( () => {
const increaseCount = () => {
clearTimeout(timerID)
setCount(count + 1)
}
const timerID = setTimeout(increaseCount, 1000)
return () => {
clearTimeout(timerID)
}
}, [count]) // count 상태가 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
1초마다 자동으로 숫자를 카운팅하는 로직이다. 리액트 훅( state hook, effect hook) 을 이용하여 관련된 코드를 분리하였다. count 를 초기화하고 업데이트하고, 카운팅 타이머를 설정/해제하는 일련의 과정을 하나의 로직으로 묶었다.
// 1초마다 자동으로 랜덤한 숫자를 보여주는 로직
const [number, setNumber] = useState(0)
useEffect( () => {
const pickRandomNumber = () => {
clearTimeout(timerID)
const randNum = Math.floor(Math.random()*100)
setNumber(randNum)
}
const timerID = setTimeout(pickRandomNumber, 1000)
return () => {
clearTimeout(timerID)
}
}, [count]) // count 상태가 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
1초마다 자동으로 랜덤한 숫자를 선택하는 로직이다. 리액트 훅( state hook, effect hook) 을 이용하여 관련된 코드를 분리하였다. number 를 초기화하고 업데이트하고, 랜덤숫자를 선택하는 타이머를 설정/해제하는 일련의 과정을 하나의 로직으로 묶었다.
의존성 배열에 number 로 설정하게 되면 랜덤숫자가 동일한 경우 해당 useEffect 함수가 동작하지 않고 멈추게 된다. 왜냐하면 useEffect 함수는 의존성 배열에 넣어준 값이 변경되었을때만 재실행되기 때문이다. 그렇게 되면 타이머가 중간에 멈춰서 더이상 랜덤한 숫자를 뽑아주지 않는다. 그러므로 1초마다 항상 변경되는 count 값을 의존성 배열에 설정해서 항상 랜덤한 숫자를 뽑아주도록 한다.
* 리액트 훅 로직 재사용하기 - Custom Hook
1초마다 랜덤한 이미지를 보여주고, 동시에 1초마다 랜덤한 영단어를 보여주는 컴포넌트를 만들고 싶다고 가정해보자!
const animalData = [
{
title: '고양이',
src: ''
},
{
title: '강아지',
src: ''
},
{
title: '햄스터',
src: ''
},
{
title: '돼지',
src: ''
},
{
title: '고슴도치',
src: ''
},
]
export default animalData;
animalData.js 파일을 위와 같이 생성하자! 복사 붙여넣기 하자!
const dictionaryData = [
{
word: 'apple',
meaning: '사과'
},
{
word: 'before',
meaning: '이전의'
},
{
word: 'clean',
meaning: '깨끗한'
},
{
word: 'dummy',
meaning: '가짜의'
},
{
word: 'emergent',
meaning: '긴급한'
},
{
word: 'famouse',
meaning: '유명한'
},
{
word: 'give',
meaning: '(~을) 주다'
},
{
word: 'humble',
meaning: '검소한'
},
{
word: 'ingrave',
meaning: '조각하다'
},
{
word: 'jungle',
meaning: '밀림숲'
},
{
word: 'korea',
meaning: '대한민국'
},
]
export default dictionaryData;
dictionaryData.js 파일을 위와 같이 생성하자! 복사 붙여넣기 하자!
import { useState, useEffect } from 'react'
function useCounter(arrLength){
// 1초마다 자동으로 숫자를 카운팅하는 로직
const [count, setCount] = useState(0);
useEffect( () => {
const increaseCount = () => {
clearTimeout(timerID)
setCount(count + 1)
}
const timerID = setTimeout(increaseCount, 1000)
return () => {
clearTimeout(timerID)
}
}, [count]) // count 상태가 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
return count % arrLength // 배열의 길이에 따라 해당 배열의 인덱스값을 반환
}
export default useCounter
useCounter.js 파일을 위와 같이 작성해보자! 커스텀 훅은 파일이름에 접두어 use 를 붙여 생성하는 것이 관례(컨벤션)이다. 해당 로직은 1초마다 배열의 인덱스 값을 순서대로 반환한다. 배열의 길이가 5라면 0, 1, 2, 3, 4 를 순서대로 반환한다.
해당 로직은 아래와 같이 두 가지 예제에 재사용 가능하다.
import './App.css';
import useCounter from './useCounter';
import animals from './animalData'
import words from './dictionaryData'
function App() {
// 이미지 갤러리 로직
const randIndex = useCounter(animals.length)
const animal = animals[randIndex]
// 플래쉬 카드 로직
const randIndex2 = useCounter(words.length)
const dic = words[randIndex2]
return (
<div style={{width: '50%', margin: '0 auto', textAlign:'center'}}>
<h1>Image Gallary</h1>
<img src={animal.src} alt={animal.title}/>
<h2>{animal.title}</h2>
<br/>
<h1>Plash Card</h1>
<h2>{dic.word}</h2>
<h3>{dic.meaning}</h3>
</div>
)
}
export default App;
App.js 파일을 위와 같이 작성하자! 이미지 갤러리 로직과 플래쉬 카드 로직을 완전히 분리하였다. 또한, useCounter 라는 커스텀 훅을 재사용하였다. 이렇게 분리된 로직 내부에서 state (상태) 는 완전히 독립적으로 동작한다. 예를 들어 useCounter 안의 count 상태는 이미지 갤러리 로직과 플래쉬 카드 로직에서 다른 값을 가진다.
전체적인 흐름을 설명하면 아래와 같다.
- 초기 로딩시 useCounter 함수가 실행된다.
- useCounter 함수가 초기에 실행되면 count 를 0 으로 초기화하고 반환한다.
- useCounter 함수가 초기에 실행되면 useEffect 안의 콜백함수가 콜백 큐에 등록된다.
- 초기 로딩이 끝난 직후(return 문 실행후) useEffect 안의 콜백함수가 실행되면서 타이머를 설정한다.
- 타이머는 1초 후에 카운트 값을 1만큼 증가시킨다.
- count 상태가 변경되었으므로 App 컴포넌트를 새로 실행한다.
- App 컴포넌트가 다시 실행되면 useCounter 함수가 재실행되면서 변경된 count 값 1을 반환한다.
- 다시 화면이 리렌더링된 직후(return 문 실행후) 콜백큐에 등록된 콜백함수의 두번째 인자 count 를 확인하다.
- count 상태가 변경되었으므로 다시 콜백함수가 재실행되면서 타이머를 재설정한다.
- 다시 5번 순서부터 반복해서 실행된다.
* 컴포넌트 마운트와 해제시에만 Effect Hook 실행하기
import './App.css';
import React, { useState, useEffect } from 'react';
function App() {
// 서버에서 데이터를 가져오는 로직
const [movies, setMovies] = useState([])
useEffect( () => {
fetch('https://yts.mx/api/v2/list_movies.json?limit=12')
.then( res => res.json())
.then( result => {
const {data: {movies}} = result
console.log(movies)
console.log('useEffect - Movies')
setMovies(movies)
})
})
return (
<div className="App">
<h1>영화목록</h1>
{movies.map( (movie, id) => {
return(
<div key={id}>{movie.title}</div>
)
})}
</div>
);
}
export default App;
App.js 파일을 위와 같이 작성하자! 위 예제코드는 아래와 같이 서버에서 데이터를 가져오는 로직이 여러번 실행된다. 즉, 클래스형 컴포넌트의 componentDidMount 와 componentDidUpdate 메서드를 합쳐놓은 것과 같이 동작한다.
이유는 맨 처음에 setMovies 함수를 사용하여 movies 상태가 업데이트가 된 다음에 render 함수(return 반환)가 다시 호출이 된다. 이후에 render 함수가 호출이 되면 무조건 useEffect (Effect Hook) 이 재실행된다. 그럼 다시 서버에서 movies 데이터를 가져오고 다시 setMovies 함수가 호출되고 다시 render 함수가 호출되면서 다시 useEffect 함수를 호출한다. 즉, 무한루프를 돌면서 서버에 계속 접속하게 된다.
이렇게 쓸데없는 useEffect (Effect Hook)의 호출을 막고, 초기 렌더링시에 componentDidMount 와 같이 한번만 서버에서 데이터를 가져오고 싶다면 useEffect 함수의 두번째 인자로 빈 배열을 넣어주면 된다.
// 서버에서 데이터를 가져오는 로직
const [movies, setMovies] = useState([])
useEffect( () => {
fetch('https://yts.mx/api/v2/list_movies.json?limit=12')
.then( res => res.json())
.then( result => {
const {data: {movies}} = result
console.log(movies)
console.log('useEffect - Movies')
setMovies(movies)
})
}, [])
이렇게 하면 초기 렌더링시에 한번만 useEffect 의 콜백함수를 실행하게 되므로 서버에서 데이터를 한번만 가져온다.
* 클린업(clean-up)
리렌더링 직후 useEffect 의 다음 콜백함수가 실행되기 직전에 클린업에 설정된 함수가 실행된다.
mport './App.css';
import React, { useEffect, useState } from 'react';
function App() {
const [count, setCount] = useState(0);
useEffect( () => {
const increaseCount = () => {
// clearTimeout(timerID)
setCount(count + 1)
}
const timerID = setTimeout(increaseCount, 1000)
// 클린업: 리렌더링되서 특정 state, props 가 변경될때마다 콜백함수가 실행되기 직전 클린업
// 클린업 === componentWillUnmount : 의존성배열에 빈배열을 설정하기
return () => {
clearTimeout(timerID)
}
}, [count]) // count 값이 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
return (
<div className="App">
<h1>Increase Count automatically !</h1>
<h2>Count: {count}</h2>
</div>
);
}
export default App;
state 이전값은 클린업 콜백함수에서 조회가 가능하다.
import './App.css';
import React, { useEffect, useState } from 'react';
function App() {
const [count, setCount] = useState(0);
useEffect( () => {
console.log("현재값: ", count)
const increaseCount = () => {
// clearTimeout(timerID)
setCount(count + 1)
}
const timerID = setTimeout(increaseCount, 1000)
// 클린업: 리렌더링되서 특정 state, props 가 변경될때마다 콜백함수가 실행되기 직전 클린업
// 클린업 === componentWillUnmount : 의존성배열에 빈배열을 설정하기
return () => {
console.log("이전값: ", count)
clearTimeout(timerID)
}
}, [count]) // count 값이 업데이트될때마다 최신 count 값을 이용하여 콜백함수를 실행함
return (
<div className="App">
<h1>Increase Count automatically !</h1>
<h2>Count: {count}</h2>
</div>
);
}
export default App;
첫번째 콜백이 실행되고, 클린업이 되고, 두번째 콜백이 실행된다. 이후 반복된다.
* 리액트 훅 연습과제 0
모달창을 열고 닫는 아래 예제코드를 리액트 훅을 사용하여 함수형 컴포넌트로 변경해보자!
import './App.css';
import Modal from './Modal';
import Button from './Button';
import { Component } from 'react';
class App extends Component {
state = {
open: false
}
openModal = () => {
this.setState({ open: true })
}
closeModal = () => {
this.setState({ open: false })
}
render(){
const { open } = this.state
const { openModal, closeModal} = this
return (
<div className="App">
<Button handleClick={openModal}>Add Todo</Button>
<Modal open={open}>
<div className="header">You want to add new todo ?</div>
<div className="body">
<label>todo name: <input type="text"></input></label><br/>
<label>todo description: <input type="text"></input></label>
</div>
<div className="footer">
<Button size="small">Add</Button>
<Button size="small" handleClick={closeModal}>Close</Button>
</div>
</Modal>
</div>
);
}
}
export default App;
* 리액트 훅 연습과제 1
import './App.css';
import React, { Component } from 'react'
class App extends Component {
state = {
count: 0
}
showUI = (cnt) => {
let ui = null;
switch(cnt){
case 0:
ui = <h1>Home</h1>
break;
case 1:
ui = <h1>About</h1>
break;
case 2:
ui = <h1>Detail</h1>
break;
default:
ui = <h1>NotFound</h1>
}
return ui
}
increase = () => {
this.setState({count: this.state.count + 1})
}
render(){
const { count } = this.state
return (
<>
{this.showUI(count)}
<button type="button" onClick={this.increase}>카운팅</button>
</>
)
}
}
export default App;
JSX 문법 수업에서 사용한 위 예제코드를 리액트 훅을 이용하여 함수형 컴포넌트에서도 동일하게 동작하게 해보자!
* 리액트 훅 연습과제 2
컴포넌트의 생명주기 예제코드에서 사진 데이터를 1초마다 순차적으로 조회하면서 웹화면에 보여주는 컴포넌트를 리액트 훅을 이용하여 함수형 컴포넌트에서도 동일하게 동작하게 해보자!
const dummyData = [
{
title: '고양이',
src: ''
},
{
title: '강아지',
src: ''
},
{
title: '햄스터',
src: ''
},
{
title: '돼지',
src: ''
},
{
title: '고슴도치',
src: ''
},
]
export default dummyData;
import './App.css';
import { Component } from 'react';
import animals from './dummyData'
class App extends Component {
state = {
count: 0
}
increaseCount = () => {
this.setState({ count: this.state.count + 1})
}
componentDidMount(){
this.countID = setInterval(
this.increaseCount
, 1000)
}
componentWillUnmount(){
clearInterval(this.countID)
}
render(){
const { count } = this.state
const animal = animals[count%animals.length]
console.log(animal)
return (
<div className="App">
<h1>Image Gallery !</h1>
<img src={animal.src} alt={animal.title}></img>
</div>
);
}
}
export default App;
클래스형 컴포넌트를 참고하자!
* 리액트 훅 연습과제 3
무비리스트를 화면에 보여주는 아래 컴포넌트를 리액트 훅을 이용하여 함수형 컴포넌트로 만들어보자!
import './App.css';
import React, { Component } from 'react';
import Movie from './Movie';
class App extends Component {
constructor(props){
super(props)
this.state = {
loading: true,
movies: []
}
}
componentDidMount(){
fetch('https://yts.mx/api/v2/list_movies.json?limit=12')
.then( res => res.json())
.then( result => {
const {data: {movies}} = result
console.log(movies)
this.setState({loading: false, movies})
})
}
render(){
const {loading, movies} = this.state
const style = {
display: 'flex',
flexWrap: 'wrap',
justifyContent: 'center',
alignItems: 'center',
width: '60%',
margin: '100px auto',
textAlign: 'center'
}
const loadingStyle = {
position: 'absolute',
left: '50%',
top:'50%',
transform: 'translate(-50%, -50%)',
fontSize: '2rem'
}
if(loading){
return (
<div style={loadingStyle}>
<h1>Loading ...</h1>
</div>
)
}else{
return (
<div style={style}>
{movies.map(movie => {
return (
<Movie
key={movie.id}
title={movie.title}
genres={movie.genres}
cover={movie.medium_cover_image}
summary={movie.summary}
></Movie>
)
})}
</div>
)
}
}
}
export default App;
import React from 'react';
function Movie({title, genres, cover, summary}){
const style = {
width: '230px',
height: '500px',
background: "white",
margin: '10px',
boxShadow: 'rgba(0, 0, 0, 0.35) 0px 5px 15px'
}
return (
<div style={style}>
<img src={cover} alt={title}></img>
<h3>{title}</h3>
<h4>{genres.join(" ")}</h4>
{/* <p>{summary}</p> */}
</div>
)
}
export default Movie;
* 리액트 훅 연습과제 4
1초마다 한번씩 자동으로 로또번호를 생성하는 다음의 코드를 리액트 훅을 사용하여 함수형 컴포넌트로 만돌어보자!
import './App.css';
import React, { Component } from 'react'
// import animals from './dummyData'
class App extends Component {
state = {
numbers: ''
}
pickRandomNumber = (min, max) => { return Math.floor( Math.random() * (max-min+1) ) + min }
isDuplicated = (numbers, picked) => {
return numbers.find(num => num === picked)
}
getLottoNum = (numbers) => {
// console.log("length: ", numbers)
const picked = this.pickRandomNumber(1, 45)
const duplicatedNum = this.isDuplicated(numbers, picked) // 중복체크
if(duplicatedNum){
console.log('duplicated ...', duplicatedNum)
this.getLottoNum(numbers) // 로또배열에 랜덤으로 뽑은 숫자가 이미 존재하면 재귀적으로 다시 숫자를 뽑음
}else{
numbers.push(picked)
}
}
showRandomNumber = () => {
const numbers = [] // 로또번호 배열
while(numbers.length < 6){
this.getLottoNum(numbers)
}
this.setState({ numbers: numbers.join(' ')})
}
// 초기에 웹화면이 렌더링되었을때 타이머를 설정함
componentDidMount(){
this.countID = setInterval(this.showRandomNumber, 1000)
}
// 사용자가 웹화면을 벗어나면 타이머를 해제함
componentWillUnmount(){
clearInterval(this.countID)
}
render(){
const { numbers } = this.state
return (
<div className='App'>
<h1>로또번호 자동 생성기</h1>
<h1>{numbers}</h1>
</div>
)
}
}
export default App;
* 리액트 훅 연습과제 5
아래 사전 검색 서비스 (리액트 버전)을 리액트 훅을 사용하여 함수형 컴포넌트로 만들어보자!
import './Word.css'
function Word({ r_link, r_word, r_hanja, r_des}){
return (
<div className="item">
<div className="word">
<a href={r_link}>{r_word} {r_hanja}
</a>
</div>
<p className="description">{r_des}</p>
</div>
)
}
export default Word;
Word.js 파일은 위와 같다.
.item {
width: 100%;
margin-bottom: 10px;
display: inline-block; /* 컬럼 짤림 방지*/
background: lightgreen;
}
Word.css 파일은 위와 같다.
import './App.css';
import React, { Component } from 'react';
import Word from './Word'
class App extends Component {
state = {
words: []
}
componentDidMount(){
const BASE_URL = 'https://dictionary-search.herokuapp.com/api/words'
fetch(BASE_URL, {
headers: {
"Content-Type": "application/json",
// "Access-Control-Allow-Origin": "*" // 이 코드 때문에 CORS 에러가 발생한것임. 이 코드 주석처리하면 프론트엔드에서 곧바로 외부 API 접근가능하다. (프록시나 서버가 필요없음)
}
})
.then( res => res.json())
.then( data => {
console.log(data)
const {words} = data;
this.setState({ words })
})
}
render(){
const { words } = this.state
return (
<div className="App">
{words.map( (word, id) => {
return (
<Word
key={id}
r_link={word.r_link}
r_word={word.r_word}
r_hanja={word.r_hanja}
r_des={word.r_des}
></Word>
)
})}
</div>
);
}
}
export default App;
App.js 파일은 위와 같다.
.App{
width: 60%;
columns: 2;
margin: 50px auto;
}
App.css 파일은 위와 같다.
* 리액트 훅 연습과제 6
이벤트 처리하기 예제로 주어진 영단어 삭제하기 코드를 리액트 훅을 사용하여 함수형 컴포넌트로 만들어보자!
import './App.css'; import React, { Component } from 'react';
import words from './dictionaryData'
import Button from './Button'
class App extends Component {
state = {
words: words
}
handleRemove = (id, e) => {
const word = e.target.previousSibling.innerText
console.log(word)
console.log(id)
alert(`You want to delete word - ${word}?`)
const words = this.state.words.filter( (w, index) => index !== id ) // 제거하려는 단어의 id 와 일치하는 요소만 걸러냄
this.setState({words})
}
render(){
const wordStyle = { display: 'flex', alignItems: 'center', justifyContent: 'center' }
const {words} = this.state
return (
<div>
<h1 style={{textAlign:'center'}}>Word List</h1>
{words.map( (w, id) => {
return (
<div key={id} style={wordStyle}>
<h2>{w.word}</h2>
<Button size="small" type="button" handleClick={(e) => this.handleRemove(id, e)}>DELETE</Button>
</div>
)
})}
</div>
);
}
}
export default App;
const dictionaryData = [
{
word: 'apple',
meaning: '사과'
},
{
word: 'before',
meaning: '이전의'
},
{
word: 'clean',
meaning: '깨끗한'
},
{
word: 'dummy',
meaning: '가짜의'
},
{
word: 'emergent',
meaning: '긴급한'
},
{
word: 'famouse',
meaning: '유명한'
},
{
word: 'give',
meaning: '(~을) 주다'
},
{
word: 'humble',
meaning: '검소한'
},
{
word: 'ingrave',
meaning: '조각하다'
},
{
word: 'jungle',
meaning: '밀림숲'
},
{
word: 'korea',
meaning: '대한민국'
},
]
export default dictionaryData;
dictionaryData.js 파일은 위와 같다.
* 리액트 훅 연습과제 7
이벤트 처리하기 예제로 주어진 이미지 뷰어 코드를 리액트 훅을 사용하여 함수형 컴포넌트로 만들어보자!
import './App.css';
import React, { Component } from 'react';
import images from './imageData'
import Button from './Button'
class App extends Component {
state = {
index: 0
}
decreaseIndex = () => {
const nextIndex = this.state.index - 1
this.setState({index: (nextIndex < 0) ? images.length - 1 : nextIndex})
}
increaseIndex = () => {
const nextIndex = this.state.index + 1
this.setState({index: (nextIndex > images.length - 1) ? 0 : nextIndex})
}
render(){
const { index } = this.state
const { increaseIndex, decreaseIndex } = this
const path = images[index].src
const title = images[index].title
return (
<div className="App">
<div className="img-container">
<img src={path} alt={title}/>
</div>
<div className="control-btns">
<Button handleClick={decreaseIndex}>Prev</Button>
<Button handleClick={increaseIndex}>Next</Button>
</div>
</div>
);
}
}
export default App;
const imageData = [
{
title: '고양이',
src: ''
},
{
title: '강아지',
src: ''
},
{
title: '햄스터',
src: ''
},
{
title: '돼지',
src: ''
},
{
title: '고슴도치',
src: ''
},
]
export default imageData;
imageData.js 파일은 위와 같다.
* 리액트 훅 연습과제 8
사용자 입력을 처리하는 아래 예제코드를 리액트 훅을 사용하여 함수형 컴포넌트로 만들어보자!
import './App.css';
import React, { Component } from 'react';
import Button from './Button'
class App extends Component {
state = {
id: '',
password: ''
}
handleChange = (e) => {
const { name, value } = e.target
console.log(name, value)
this.setState({ [name]: value}) // 주석처리하면 사용자 입력이 되지 않음
}
login = (e) => {
e.preventDefault() // 새로고침 방지
console.log('login')
}
render(){
const { id, password } = this.state
return (
<div className="App">
<form>
<label>ID <input type="text" placeholder="TYPE YOUR ID ..." name="id" value={id} onChange={this.handleChange}></input></label><br/><br/>
<label>PASSWORD <input type="password" placeholder="TYPE YOUR PASSWORD ..." name="password" value={password} onChange={this.handleChange}></input></label>
<div className="login-btn"><Button handleClick={this.login}>Login</Button></div>
</form>
</div>
);
}
}
export default App;
.App{
width: 300px;
clear: both;
margin: 200px auto;
}
.App label{
color:lightgreen;
font-weight: bold;
margin-left: 5px;
}
.App input {
width: 100%;
height: 30px;
clear: both;
border: 2px solid lightgreen;
border-radius: 15px;
outline: none;
padding: 10px;
color: lightgreen;
font-weight: bold;
font-size: 1rem;
}
.App input::placeholder{
color: lightgreen;
}
.login-btn{
width: 320px;
display: flex;
justify-content: flex-end;
}
App.css 파일은 위와 같다.
* 리액트 훅 연습과제 9
파일 입력을 처리하는 아래 예제코드를 리액트 훅을 사용하여 함수형 컴포넌트로 만들어보자!
import './App.css';
import React, { Component } from 'react';
import Button from './Button'
class App extends Component {
constructor(props){
super(props)
this.state = {
fileName: '',
imgSrc: ''
}
this.fileInput = React.createRef() // ref 생성하기
}
isValid = (type) => {
return type === 'image'
}
handleChange = (e) => {
console.log(e.target.files[0])
const file = e.target.files[0]
const imgSrc = URL.createObjectURL(file)
if(this.isValid(file.type.split('/')[0])){
this.setState({ fileName: file.name, imgSrc })
}else{
this.setState({ fileName: 'File is not valid type !', imgSrc: ''})
}
}
openFileWindow = () => {
this.fileInput.current.click() // ref 사용하기
}
render(){
const { fileName, imgSrc } = this.state
return (
<div className="App">
<h1>{fileName}</h1>
{imgSrc !== '' && <img src={imgSrc} alt="preview-img" width="300px" height="400px"></img> }
<input className="Upload" type="file" onChange={this.handleChange} ref={this.fileInput} accept="image/*"></input>
<Button handleClick={this.openFileWindow}>Upload</Button>
</div>
);
}
}
export default App;
.App{ width: 30%; margin: 200px auto; } .Upload{ display: none; }
App.css 파일은 위와 같다.
'프론트엔드 > React' 카테고리의 다른 글
리액트 기초이론 5 - 컴포넌트 스타일링 2 - SASS (0) | 2022.02.28 |
---|---|
리액트 기초이론 7 - 이벤트(Event) 처리하기 (0) | 2021.11.07 |
리액트 기초이론 8 - 리액트 라우터 (0) | 2021.10.22 |
리액트 기초이론 6 - 요소 참조(ref) (0) | 2021.10.22 |
리액트 기초이론 5 - 컴포넌트 스타일링 (0) | 2021.10.22 |