<Router>
components of React Router v4?push()
and replace()
methods of history
?history.push
method in React Router v4?mapStateToProps()
and mapDispatchToProps()
?connect()
from React Redux?at
symbol in the Redux connect decorator?mapDispatchToProps()
?ownProps
parameter in mapStateToProps()
and mapDispatchToProps()
?call()
and put()
in redux-saga?redux-saga
and redux-thunk
?React Router는 리액트 최상단에 있는 강력한 라우팅 라이브러리로, 페이지에 보여주는 내용과 URL사이에 동기화를 유지해주고, 애플리케이션에 새로운 화면과 흐름을 추가할 수 있도록 도와준다.
React router는 history라이브러리를 감싼 래퍼로, 브라우저의 window.history
와 상호작용하고, 브라우저 및 해쉬의 히스토리를 다룬다. 또한 모바일 앱 개발 (React Native) 및 Node의 unit testing처럼 global histroy가 없는 환경에 유용한 메모리 히스토리를 제공한다.
<Router>
components of React Router v4?v4는 새로운 3개의 <Router>
컴포넌트를 제공한다.
<BrowserRouter>
<HashRouter>
<MemoryRouter>
위 컴포넌트는 각각 브라우저, 해쉬, 메모리 히스토리 인스턴스를 만들어준다. React Router v4는 Router Object의 context를 통해, history 인스턴스의 속성과 메소드를 활용할 수 있게 해준다.
push()
and replace()
methods of history
?히스토리 인스턴스에는 네비게이션 목적으로 두개의 메소드를 제공한다.
push()
replace()
만약 히스토리가 방문했던 곳들의 배열이라고 생각한다면, push()
가 그 역할을 할 것이고, 현재 위치를 덮어쓰는 느낌을 원한다면 replace()
가 맞을 것이다.
Component 내에서 프로그래밍으로 라우팅/네비게이팅 하는 방법에는 3가지가 있다.
withRouter()
를 쓰는법
HOF의 withRouter()
는 컴포넌트의 prop에 히스토리 오브젝트를 인젝트 한다. 이 오브젝트는 push()
replace()
를 제공하여 context의 사용을 피하게 해준다.import { withRouter } from 'react-router-dom' // this also works with 'react-router-native'
const Button = withRouter(({ history }) => (
<button
type="button"
onClick={() => {
history.push('/new-location')
}}
>
{'Click Me!'}
</button>
))
<Route>
컴포넌트와 render props 패턴을 사용하는 법
<Route>
는 withRouter()
와 같은 props를 넘기므로, history prop을 통해 histoy 메서드에 접근할 수 있을 것이다.import { Route } from 'react-router-dom'
const Button = () => (
<Route
render={({ history }) => (
<button
type="button"
onClick={() => {
history.push('/new-location')
}}
>
{'Click Me!'}
</button>
)}
/>
)
const Button = (props, context) => (
<button
type="button"
onClick={() => {
context.history.push('/new-location')
}}
>
{'Click Me!'}
</button>
)
Button.contextTypes = {
history: React.PropTypes.shape({
push: React.PropTypes.func.isRequired,
}),
}
수년간 다른 구현 지원에 대한 사용자들의 많은 요청 때문에, React Router v4에서는 query string을 parsing 하는 방법은 사라졌다. 이는 유저가 원하는 대로 구현할 수 있는 자유도를 주었다. 추천하는 방법은, query string 라이브러리를 사용하는 것이다.
const queryString = require('query-string')
const parsed = queryString.parse(props.location.search)
native 방식을 선호한다면 URLSearchParam
을 사용할 수도 있다.
const params = new URLSearchParams(props.location.search)
const foo = params.get('name')
다만 IE11에서는 폴리필이 필요하다.
Route는 <Switch>
블록으로 감싸줘야 하는데, 왜냐하면 <Switch>
는 라우트를 베타적으로 감싸기 때문이다. 먼저 Switch
를 임포트 해야 한다.
import { Switch, Router, Route } from 'react-router'
그리고 route를 <Switch>
블록에 넣어햐 한다.
<Router>
<Switch> <Route {/* ... */} /> <Route {/* ... */} /> </Switch>
</Router>
history.push
method in React Router v4?history 객체에 props를 보낼 수 있다.
this.props.history.push({
pathname: '/template',
search: '?name=sudheer',
state: { detail: response.data },
})
search
속성은 push()
에서 query param을 보낼 때 사용된다.
<Switch>
는 첫번째로 일치하는 <Route>
를 렌더링한다. path가 없는 route는 항상 매치하게 되어 있다. 따라서, path를 제거한 route를 하나 추가하면 된다.
<Switch>
<Route exact path="/" component={Home} />
<Route path="/user" component={User} />
<Route component={NotFound} />
</Switch>
import { createBrowserHistory } from 'history'
export default createBrowserHistory({
/* pass a configuration object here if needed */
})
<Router>
컴포넌트를 쓴다. 위에서 만든 history.js
를 index.js
에 임포트 한다.import { Router } from 'react-router-dom'
import history from './history'
import App from './App'
ReactDOM.render(
<Router history={history}>
<App />
</Router>,
holder,
)
// some-other-file.js
import history from './history'
history.push('/go-here')
react-router
sms <Redirect>
컴포넌트를 제공한다. <Redirect>
를 렌더링 하면 새로운 위치로 이동하게 된다. 서버사이드 리다이렉트와 마찬가지로, 새로운 위치는 현재 히스토리 스택에 있는 현재 위치를 덮어쓰게 된다.
import React, { Component } from 'react'
import { Redirect } from 'react-router'
export default class LoginComponent extends Component {
render() {
if (this.state.isLoggedIn === true) {
return <Redirect to="/your/redirect/page" />
} else {
return <div>{'Login Please'}</div>
}
}
}
React Intl string, dates, numbers, 복수 표현 등을 다국어로 포맷팅할 수 있는 컴포넌트와 API를 제공한다. React Intl는 components 와 API 를 바탕으로 Reac를 바인딩하는 FormatJS 의 일부분이다.
string, number, date를 포맷팅하는 방법은 react 컴포넌트 또는 api를 사용하는 두가지 방법이 있다.
<FormattedMessage
id={'account'}
defaultMessage={'The amount is less than minimum balance.'}
/>
const messages = defineMessages({
accountMessage: {
id: 'account',
defaultMessage: 'The amount is less than minimum balance.',
},
})
formatMessage(messages.accountMessage)
<FormattedMessage>
as placeholder using React Intl?<Formatted... />
컴포넌트는 plain text가 아닌 elements를 반환하므로, placeholder, alt text처럼 string이 필요한 곳에는 쓸 수 없다. 따라서 여기에서는 formatMessage()
를 사용해야한다. higher-order component인 injectIntl()을 사용하여, 컴포넌트에 intl 객체를 주입하고, 객체에서 사용할 수 있는 formatMessage()
를 사용하여 message를 포맷팅할 수 있다.
import React from 'react'
import { injectIntl, intlShape } from 'react-intl'
const MyComponent = ({ intl }) => {
const placeholder = intl.formatMessage({ id: 'messageId' })
return <input placeholder={placeholder} />
}
MyComponent.propTypes = {
intl: intlShape.isRequired,
}
export default injectIntl(MyComponent)
어느 애플리케이션에서든 injectIntl()
를 사용하면 현재 로케일을 얻을 수 있다.
higher-order 컴포넌트 injectIntl()
는 컴포넌트의 props에 formatDate()
메서드를 제공한다. 이 메서드는 내부적으로 FormattedDate
인스턴스를 활용하고, 이는 포맷된 날짜를 string으로 제공한다.
import { injectIntl, intlShape } from 'react-intl'
const stringDate = this.props.intl.formatDate(date, {
year: 'numeric',
month: 'numeric',
day: 'numeric',
})
const MyComponent = ({ intl }) => (
<div>{`The formatted date is ${stringDate}`}</div>
)
MyComponent.propTypes = {
intl: intlShape.isRequired,
}
export default injectIntl(MyComponent)
Shallow rendering
는 React에서 유닛테스트 케이스를 작성할 때 유용하다. 이는 컴포넌트를 한단계 더 깊이 렌더링하며, 렌더링되지 않은 하위 컴포넌트에 대한 고민 ㅇ벗이 렌더링 메서드가 반환하는 것에 대해 asset를 수행할 수 있다.
function MyComponent() {
return (
<div>
<span className={'heading'}>{'Title'}</span>
<span className={'description'}>{'Description'}</span>
</div>
)
}
import ShallowRenderer from 'react-test-renderer/shallow'
const renderer = new ShallowRenderer()
renderer.render(<MyComponent />)
const result = renderer.getRenderOutput()
expect(result.type).toBe('div')
expect(result.props.children).toEqual([
<span className={'heading'}>{'Title'}</span>,
<span className={'description'}>{'Description'}</span>,
])
TestRenderer
package in React?TestRenderer
패키지는 component 를 DOM 또는 Native mobile 환경에 의존없이 순수 Javascript Object 로 렌더링 할 수 있는 renderer 를 제공한다. 이 패키지를 사용하면 브라우저 또는 jsdom 의 사용없이 ReactDOM 또는 React Native 에서 렌더링 되는 플랫폼의 뷰 계층구조 (DOM 트리와 유사) 의 스냅샷을 쉽게 가져올 수 있다.
import TestRenderer from 'react-test-renderer'
const Link = ({ page, children }) => <a href={page}>{children}</a>
const testRenderer = TestRenderer.create(
<Link page={'https://www.facebook.com/'}>{'Facebook'}</Link>,
)
console.log(testRenderer.toJSON())
// {
// type: 'a',
// props: { href: 'https://www.facebook.com/' },
// children: [ 'Facebook' ]
// }
ReactTestUtils
는 유닛테스트를 목적으로 DOM을 조작할 수 있는 with-addons
패키지를 제공한다.
Jest는 페이스북이 만든 자바스크립트 유닛테스트 프레임워크로, Jasmine을 기반으로 만들어 졌으며 자동 mock 생성, jsdom
환경 제공 등의 기능을 제공한다. 컴포넌트를 테스트 하는데 사용 된다.
Jasmine보다 Jest가 더 좋은 점은
두 숫자를 더하는 sum.js
를 작성한다.
const sum = (a, b) => a + b
export default sum
테스트를 수행하는 sum.test.js
를 작성
import sum from './sum'
test('adds 1 + 2 to equal 3', () => {
expect(sum(1, 2)).toBe(3)
})
package.json
에 테스트를 실행하는 코드 추가
{
"scripts": {
"test": "jest"
}
}
yarn test
npm test
로 테스트 실행 및 결과 확인
$ yarn test
PASS ./sum.test.js
✓ adds 1 + 2 to equal 3 (2ms)
Flux는 애플리케이션 디자인 패러다임으로, 전통적인 모델인 MVC pattern을 대체하기 위해 나왔다. Flux는 프레임워크나 라이브러리가 아닌, React와 양방향 데이터 흐름을 기반으로 하는 새로운 아키텍쳐다. 페이스북이 React를 사용할 때 내부적으로 이 패턴을 활용한다.
dispatcher, sotres, views 컴포넌트 사이 작업흐름은 아래처럼 input과 output이 구별되어 나타난다.
Redux는 flux 디자인 패턴을 기반으로 한 자바스크립트 앱의 예측가능한 state container다. Redux는 React또는 다른 어떤 뷰 라이브러리와 함께 사용할 수 있다. Redux는 크기가 매우 작고 (2kb), 다른 디펜던시를 갖고 있지 않다.
Redux는 다음 세가지 기본 원칙을 가지고 있다.
Flux와 비교했을 때, Redux는 몇가지 단점을 가지고 있다.
redux-immutable-state-invariant
나 Immutable.js
를 활용하거나, 팀원들에게 변이 없는 코드에 대해 방법론을 확산해야 한다.mapStateToProps()
and mapDispatchToProps()
?mapStateToProps()
는 컴포넌트에서 다른 컴포넌트에 의해 업데이트된 state를 가져올수 있도록 도와주는 유틸리티다.
const mapStateToProps = (state) => {
return {
todos: getVisibleTodos(state.todos, state.visibilityFilter),
}
}
mapDispatchToProps()
는 컴포넌트가 이벤트를 발생시킬 수 있도록 도와주는 유틸리티다. (이 이벤트는 애플리케이션의 state에 변화를 가져올 수 있음)
const mapDispatchToProps = (dispatch) => {
return {
onTodoClick: (id) => {
dispatch(toggleTodo(id))
},
}
}
mapDispatchToProps
에서는 항상 객체를 파라미터로 보내기를 권장한다.
Redux는 (…args) => dispatch(onTodoClick(…args))
와 같은 형태의 다른 함수로 감싸고, 이렇게 감싼 함수를 컴포넌트의 prop로 전달한다.
const mapDispatchToProps = {
onTodoClick,
}
Reducer안에서 액션을 보내는 것은 안티패턴이다. Reducer는 사이드이펙트를 최소화 하기 위하여, 단순히 액션에 대한 처리와 새로운 state를 가진 object를 반환하기만 해야 한다. Reducer내에서 리스너를 달고, 액션을 보내는 것은 다른 액션과 연쇄작용을 일으킬 수도 있으며, 사이드 이펙트를 야기할 수도 있다.
createStore()
로 만들어진 모듈을 export 하면 된다. 그리고 global 객체인 window를 사용해서는 안된다.
store = createStore(myReducer)
export default store
두 라이브러리는 목적부터 완전히 다르지만, 약간의 비슷한점을 가지고있다.
Redux는 애플리케이션 전반에서 state를 관리할 수 있게 도와주는 툴이다. 이는 보통 UI 아키텍쳐에서 ㅁ낳이 사용된다. Angular의 대체재라고 볼 수 있다. 반면 Rxjs는 반응형 프로그래밍 라이브러리다. RxJS는 자바스크립트에서 비동기 작업을 수행하기 위해 사용된다. Promise의 대체재라고 볼 수있다. Redux는 Store가 반응형이기 때문에 반응형 패러다임을 사용한다. Store는 액션을 어느정도 거리에서 관찰하다가, 스스로 변화한다. RxJS 또한 반응형 패러다임을 사용하는 반면, 아키텍쳐를 제공하지 않고 Observable 과 같은 블록을 제공한다.
componentDidMount()
와 render()
메서드에서 데이터를 확인하는 액션을 전닥ㄹ할 수 있고 데이터를 확인할 수 있다.
class App extends Component {
componentDidMount() {
this.props.fetchData()
}
render() {
return this.props.isLoaded ? (
<div>{'Loaded'}</div>
) : (
<div>{'Not Loaded'}</div>
)
}
}
const mapStateToProps = (state) => ({
isLoaded: state.isLoaded,
})
const mapDispatchToProps = { fetchData }
export default connect(mapStateToProps, mapDispatchToProps)(App)
connect()
from React Redux?container에서 store를 사용하기 위해서는 아래 두단계를 따라야 한다.
mapStateToProps()
를 사용: state의 값을 props에서 지정한 store에 맵핑시킨다.mapStateToProps()
에 의해 리턴되는 객체들은 컨테이너와 연결된다. 이를 react-redux
의 connect
로 import 할 수 있다.import React from 'react'
import { connect } from 'react-redux'
class App extends React.Component {
render() {
return <div>{this.props.containerData}</div>
}
}
function mapStateToProps(state) {
return { containerData: state.data }
}
export default connect(mapStateToProps)(App)
combineReducers()
로 생성된 reducer 에게 action 을 위임하도록 application 단에서 root reducer 를 작성해야 한다.
예를 들어, USER_LOGOUT
액션에 초기 state값을 리턴하는 rootReducer()
를 예로 들어보자. 알다시피, reducer는 action에 상관없이 첫 번째 매개변수가 undefined로 호출된다면, 초기 상태값을 반환한다.
const appReducer = combineReducers({
/* your app's top-level reducers */
})
const rootReducer = (state, action) => {
if (action.type === 'USER_LOGOUT') {
state = undefined
}
return appReducer(state, action)
}
redux-persist
를 사용하는 경우, 스토리지를 비워야 할 수도 있다. redux-persist
에서는 스토리지 안진에 있는 state의 사본을 보관해둔다. 먼저, 적절한 스토리지 엔진을 임포트 한다음, 상태를 undefined로 설정하기 전에 storage state key를 비워주어야 한다.
at
symbol in the Redux connect decorator?@
는 자바스크립트에서 데코레이터를 나타낼 때 쓰는 표현식이다. 데코레이터는 class와 속성에 주석을 달고, 이를 수정할 수 있게 해준다.
데코레이터가 없는 redux를 예로 들어보자.
import React from 'react'
import * as actionCreators from './actionCreators'
import { bindActionCreators } from 'redux'
import { connect } from 'react-redux'
function mapStateToProps(state) {
return { todos: state.todos }
}
function mapDispatchToProps(dispatch) {
return { actions: bindActionCreators(actionCreators, dispatch) }
}
class MyApp extends React.Component {
// ...define your main app here
}
export default connect(mapStateToProps, mapDispatchToProps)(MyApp)
import React from 'react'
import * as actionCreators from './actionCreators'
import { bindActionCreators } from 'redux'
import { connect } from 'react-redux'
function mapStateToProps(state) {
return { todos: state.todos }
}
function mapDispatchToProps(dispatch) {
return { actions: bindActionCreators(actionCreators, dispatch) }
}
@connect(mapStateToProps, mapDispatchToProps)
export default class MyApp extends React.Component {
// ...define your main app here
}
위 예제는 데코레이터를 사용한 것을 제외하고는 비슷하다. 데코레이터는 아직 자바스크립트 런타임에 구현되어 있지 않다. 여전히 실험적인 내용이기 때문에 수정될 여지가 있다. 바벨을 사용하면 이 데코레이터를 쓸 수 있다.
Context는 애플리케이션에서 다이렉트로 사용할 수 있으며, 깊게 중첩된 컴포넌트에 데이터를 전달하는데 유용하다. 반면 Redux는 훨씬 더 강력하며, Context API가 제공하지 않는 기능을 제공한다. 또한, React Redux 는 내부적으로 context를 활용하지만, public api에 공개하지는 않는다.
Reducers 는 항상 모든 이전과 현재의 action을 기반으로한 상태값을 반환한다. Redux reducer 가 호출 될 때 마다 상태와 액션이 파라미터로 전달된다. 상태는 action 에 따라 감소되거나 누적되어 다음 상태를 반환한다. 최종 상태를 얻기 위한 action을 실행하는데 action 단위와 store 의 초기 상태 값을 줄일 수 있다.
비동기 액션을 허용하는 미들웨어인 redux-thunk
를 사용하면 가능하다.
export function fetchAccount(id) {
return (dispatch) => {
dispatch(setLoadingAccountState()) // Show a loading spinner
fetch(`/account/${id}`, (response) => {
dispatch(doneFetchingAccount()) // Hide loading spinner
if (response.status === 200) {
dispatch(setAccount(response.json)) // Use a normal function to set the received state
} else {
dispatch(someError)
}
})
}
}
function setAccount(data) {
return { type: 'SET_Account', data: data }
}
Redux Store 에서는 Data를 저장하고, 컴포넌트 내부에서는 UI 에 관련된 상태들을 저장한다.
컴포넌트에서 스토어에 접근하는 좋은 방법은 connect()
함수를 이용하는 것이다. 이 함수는 이미 존재하는 컴포넌트를 감싸 새로운 컴포넌트를 만든다. 이러한 방식을 HOC(Higher Order Component)라고 하는데, 이는 리액트에서 컴포넌트의 기능을 확장할 때 주로 사용한다. 이 방법은 상태와 action 생성자를 컴포넌트에 매핑하고, store가 업데이트 되면 자동적으로 컴포넌트에 state와 action 생성자를 전달 할 수 있도록 해준다.
conenct를 사용한 <FilterLink>
component예제를 아래에서 살펴보자.
import { connect } from 'react-redux'
import { setVisibilityFilter } from '../actions'
import Link from '../components/Link'
const mapStateToProps = (state, ownProps) => ({
active: ownProps.filter === state.visibilityFilter,
})
const mapDispatchToProps = (dispatch, ownProps) => ({
onClick: () => dispatch(setVisibilityFilter(ownProps.filter)),
})
const FilterLink = connect(mapStateToProps, mapDispatchToProps)(Link)
export default FilterLink
이미 성능최적화가 되어 있고, 버그를 발생할 여지도 적기 때문에 개발자들은 context api로 바로 스토어에 접근하는 것 보다는 connect()
를 사용하는 것을 더 선호한다.
Component
는 애플리케이션의 일부분을 표시하는 함수 또는 클래스 컴포넌트를 의미한다.
Container
는 비공식적인 용어로, Redux Store와 연결된 컴포넌트를 지칭한다. Container 는 Redux 의 state update 와 action 을 구독하며, DOM element 를 렌더링하지 않는다. 이러한 rendering응ㄴ 하위 component 들에게 위임한다.
상수를 사용하면 IDE를 사용할 때 프로젝트 전체에서 특정한 기능의 모든 사용내역을 쉽게 찾을 수 있다. 또한 오타로 인한 버그도 방지할 수 있다. 오타가 난다면 즉시 ReferenceError
를 낸다.
일반적으로 constant.js
또는 actionTypes.js
에 저장한다.
export const ADD_TODO = 'ADD_TODO'
export const DELETE_TODO = 'DELETE_TODO'
export const EDIT_TODO = 'EDIT_TODO'
export const COMPLETE_TODO = 'COMPLETE_TODO'
export const COMPLETE_ALL = 'COMPLETE_ALL'
export const CLEAR_COMPLETED = 'CLEAR_COMPLETED'
이 파일은 두 군데에서 사용된다.
import { ADD_TODO } from './actionTypes'
export function addTodo(text) {
return { type: ADD_TODO, text }
}
import { ADD_TODO } from './actionTypes'
export default (state = [], action) => {
switch (action.type) {
case ADD_TODO:
return [
...state,
{
text: action.text,
completed: false,
},
]
default:
return state
}
}
mapDispatchToProps()
?mapDispatchToProps()
안에서 dispatch() 를 사용하여 action creators를 바인딩하는 방법은 몇가지가 있다.
const mapDispatchToProps = (dispatch) => ({
action: () => dispatch(action()),
})
const mapDispatchToProps = (dispatch) => ({
action: bindActionCreators(action, dispatch),
})
const mapDispatchToProps = { action }
ownProps
parameter in mapStateToProps()
and mapDispatchToProps()
?ownProps
파라미터가 명시되어 있다면, React Redux는 component로 전달된 props를 연결된 함수로 전달한다. 그래서 만약 connected component를 사용한다면,
import ConnectedComponent from './containers/ConnectedComponent'
;<ConnectedComponent user={'john'} />
mapStateToProps()
와 mapDispatchToProps()
안의 ownProps
는 객체가 될 것이다.
{ "user": "john" }
이 객체를 활용하여 함수에서 무엇을 반환할지 결정할 수 있다.
대부분의 애플리케이션이 아래와 같은 상위구조 레벨을 가지고 있다.
이러한 구조는 중소규모의 애플리케이션에 적합하다.
redux-saga 는 side effects (데이터를 가져오는 비동기적인 작업이나 browser cache 에 접근하는 것등)를 React/Redux applications에서 더 쉽게 만들도록 도와주는 라이브러리다.
Saga
는 애플리케이션과 분리된 스레드와 같은것으로, 부수적인 역할을 담당하기 위한 책임을 가지고 있다. redux-saga는 redux의 미들웨어로, 메인 application 에서 Redux actions 과 함께 쓰레드를 시작, 중지, 취소 할 수 있으며 전체의 Redux application 상태에 접근할 수 있으며 Redux actions 도 전달할 수 있다.
call()
and put()
in redux-saga?call()
put()
모두 effect creator 함수다. call()
은 함수는 middleware 가 promise 를 어떻게 호출할지를 설명하는 effect 을 생성하는데 사용된다. put()
함수는 store 에 action 을 통하여 전달하도록 미들웨어에게 가르치는 effect 를 생성한다.
사용자의 데이터를 가져오는 예제를 보고 effects 가 어떻게 동작하는지 살펴보자.
function* fetchUserSaga(action) {
// `call` function accepts rest arguments, which will be passed to `api.fetchUser` function.
// Instructing middleware to call promise, it resolved value will be assigned to `userData` variable
const userData = yield call(api.fetchUser, action.userId)
// Instructing middleware to dispatch corresponding action.
yield put({
type: 'FETCH_USER_SUCCESS',
userData,
})
}
Redux Thunk 는 action 대신 함수를 반환하는 action 생성자를 작성 할 수 있는 미들웨어다. Thunk 는 action dispatch 를 지연 시키거나, 특정한 조건이 성립되는 경우에만 dispatch 하도록 할 수 있다. 내부 함수는 파라미터로로 dispatch()
getState()
를 받는다.
redux-saga
and redux-thunk
?Redux Thunk 와 Redux Saga 는 모두 side effect 를 다룬다. 대부분의 시나리오에서 Thunk 는 Promise 를 사용하여 처리하고 Saga 는 Generators 를 사용한다. Promise 는 많은 개발자들에게 친숙하기 때문에 Thunk 는 비교적 다루기 쉽고, Sagas와 Generator 는 기능은 강력한 반면에 러닝커브가 존재한다. 두 미들웨어 모두 공존 할 수 있다. Thunk 로 시작하여도 만약 Saga 가 필요하다면 도입 할 수 있다.
Redux DevTools 은 Redux 를 위한 hot reload 기능을 가진 실시간 편집이 가능한 툴이다. 액션을 다시 재현하거나 UI 를 사용자 정의에 맞게 만들 수 있다. Redux DevTools 을 프로젝트에 설치하여 사용하고 싶지 않다면 Chrome 또는 Firefox 용 Extension 사용을 고려해 볼 수 있다.
persistState()
store enhancer 을 사용하면 page reload 에서 debug session을 유지할 수 있음Selectors 는 Redux state 를 인수로받고 데이터를 반환하여 component 로 전달하는 함수다.
예를 들어, state에서 유저 상태정보를 받는다면 아래와 같이 처리할 수 있다.
const getUserData = (state) => state.user.data
Redux Form은 React와 Redux와 동시에 작동하며, React 폼 내에서 Redux의 모든 상태를 저장할 수 있다. Redux Form은 HTML5 input요소들과 사용가능하며, Material UI, React Widget, React bootstrap 과 같은 UI 프레임워크와도 동작이 가능하다.
applyMiddleware()
를 사용하면 된다. 예를 들어, applyMiddleware()
를 사용하여 redux-thunk
와 logger
를 추가할 수 있다.
import { createStore, applyMiddleware } from 'redux'
const createStoreWithMiddleware = applyMiddleware(
ReduxThunk,
logger,
)(createStore)
createStore
에 두번째 인자로 초기 state값을 넘겨주면 된다.
const rootReducer = combineReducers({
todos: todos,
visibilityFilter: visibilityFilter,
})
const initialState = {
todos: [{ id: 123, name: 'example', completed: false }],
}
const store = createStore(rootReducer, initialState)
Relay와 Redux모두 하나의 스토어를 쓴다는 점에서 같다. 가장 큰 차이점은, 서버로 붙어 받은 메시지만 릴레이 한다는 점, 그리고 상태값을 모두 GraphQL 쿼리로 받는다는 것이다. Relay는 변경된 데이터만 가져온다는 점에서 데이터를 캐싱하거나 최적화할 수 있다.
React는 자바스크립트 라이브러리로, 프론트엔드와 서버에서 동작하며, 유저인터페이스나 웹 애플리케이션을 만들기 위해 사용된다.
React Native는 네이티브 앱 컴포넌트를 컴파일하기 위한 모바일 프레임워크로, 자바스크립트 기반 React로 iOS, Android와 같은 네이티브 애플리케이션을 만들 수 있게 해준다.
React Native는 iOS나 안드로이드와 같은 시뮬레이터로만 테스트가 가능하다. expo app를 활용한다면, qr코드를 활용하여 무선 네트워크 상에서도 모바일과 컴퓨터로 싱크를 맞출 수 있다.
console.log
console.warn
을 사용할 수 있다. React Native v0.29에서는 아래 명령어로도 가능하다.
$ react-native log-ios
$ react-native log-android
Command + D
를 눌러서 웹페이지가 http://localhost:8081/debugger-ui
에서 실행되게 한다.Command + Option + I
또는 View
-> Developer
-> Developer Tools
로 크롬 개발자 도구를 띄운다.