리액트는 단일-페이지 애플리케이션과 모바일 애플리케이션의 화면처리를 도와주는 자바스크립트 라이브러리입니다. 첫 장에서는 리액트에 대해 소개하고, 개발자들이 왜 리액트를 공부해야 하는지에 대해서도 말씀드리겠습니다. 우리는 리액트 생태계에 들어가서, 간단한 첫번째 리액트 애플리케이션을 만들 것입니다. 이 과정에서 JSX와 React의 문법, 그리고 ReactDOM에 대해 알려드릴 것입니다. 여러분께서는 최신 웹 애플리케이션에서 리액트를 사용하는 방법에 대해 알게 될 것입니다.
최근 몇 년 간 단일 페이지 애플리케이션(SPA: Single Page Application)의 인기가 높아지고 있습니다. 앵귤러(Angular), 엠버(Ember) 및 백본(Backbone) 등 자바스크립트 프레임워크의 등장으로 바닐라 자바스크립트(Vanilla JavaScript : 라이브러리나 프레임워크 사용 없이 순수한 자바스크립트로 개발하는 것을 말함)나 제이쿼리(jQuery)를 사용하지 않고도 최신 웹 응용 프로그램을 구축할 수 있게 되었습니다. 앞서 말씀드린 세개의 SPA 프레임워크는 다른 프레임워크들보다 앞선 2010년과 2011년 사이에 등장했습니다. 이들 외에도 SPA 개발에 사용할 수 있는 도구는 아주 많습니다. 1세대 SPA 프레임워크들이 기업용 수준에 도달하게 되면서, 더 견고해졌습니다. 반면에 리액트는 혁신적인 라이브러리로써 Airbnb, Netflix, 그리고 Facebook 등 기술 주도 기업들에서 사용되고 있습니다.
리액트는 처음에는 페이스북의 웹 개발팀이 2013년에 뷰(View) 라이브러리로 공개했습니다. 여기서 뷰(View)란 MVC 패턴(Model–View–Controller Pattern)의 'V'를 말합니다. 리액트는 뷰 라이브러리로써, 컴포넌트들을 브라우저에 표시해줍니다. 우리가 단일 페이지 애플리케이션을 작성할 수 있는 것은 관련된 도구들이 있기 때문입니다. 1세대 프레임워크들이 많은 문제를 한꺼번에 해결하려 했다면, 리액트는 뷰 계층을 작성하는데에만 집중합니다. 구체적으로, 여기서 말하는 뷰는 조합가능한 컴포넌트 계층 구조를 말합니다. MVC 패턴에 대해 몰라도 신경쓰지 마세요. 다른 언어에서 오신 분들께 리액트의 역사적 맥락을 알려드리기 위해서 언급했을 뿐입니다.
리액트의 장점은 여러 복잡한 기능을 추가하기 전 오롯이 뷰 레이어에 집중하여 개발할 수 있다는 점입니다. 그후에 다른 요소들을 추가할 수 있습니다. SPA가 집이라면 이들은 집을 짓기 위한 벽돌과 같습니다. 이런 방식이 완성도 높은 애플리케이션을 만드는데 중요합니다. 여기에는 두가지 장점이 있는데요.
-
단계별로 하나씩 벽돌 쌓는 법을 배울 수 있습니다. 처음부터 한꺼번에 집을 짓기 위한 모든 도구를 공부하지 않아도 됩니다. 시작할 때부터 모든 벽돌을 알아야하는 프레임워크들과 다른 점입니다. 이 책은 첫번째 벽돌인 리액트에 집중할 것입니다. 그러면 다른 벽돌들도 배우기 쉬울 것입니다.
-
모든 벽돌은 상호 교환이 가능합니다. '리액트 생태계가 혁신적이다'라고 평가받는 이유도 이 때문입니다. 리액트 생태계에서는 여러 솔루션들이 서로 경쟁하며 발전하고 있습니다. 따라서 상황에 적합한 솔루션을 도입할 수 있습니다.
최신 기술을 도입해 웹 애플리케이션을 개발하고자 한다면 리액트는 가장 좋은 선택이 될 것입니다. 다시한번 말씀드리자면, 리액트는 뷰 계층만을 다룹니다. 하지만 관련된 도구들과 함께 완전히 유연하고 호환가능한 프레임워크를 이루고 있습니다. 리액트는 간결한 API, 튼튼하고 발전중인 생태계, 훌륭한 커뮤니티를 갖추고 있습니다.
만약 제가 리액트를 선택한 이유를 더 자세히 알고 싶거나, 이 장에서 다룬 내용에 대해 더 알고 싶다면, 아래 글에 더 깊은 내용이 있습니다.
이 책을 읽기 위해 웹 개발에 필요한 기초적인 HTML, CSS, 자바스크립트과 API 동작 방식에 대해 알고 있어야 합니다. 그리고 API들의 동작 방식도 알고 있어야 합니다. 이 책에서는 관련 사항을 깊게 다루게 됩니다. 또한 저는 여러분이 공식 슬랙 그룹에 가입해 동료들을 만나고 서로에게 도움을 주길 권장합니다.
실습을 진행하려면 코드에디터나 IDE(Integrated Development Environment, 통합개발환경), 터미널(terminal 또는 커맨드라인 command line)이 필요합니다. 자세한 방법은 개발 환경 설정 가이드를 참고하기 바랍니다. 추가로 이 책의 연습문제를 풀 때 깃헙(GitHub) 프로젝트에 넣을 것을 권장합니다. 관련 도구의 사용법에 대해 간단한 가이드를 작성해두었습니다. 깃헙은 훌륭한 버전 관리 도구로써, 작업 중에 문제를 일으켰을 때 수정한 부분을 확인하거나 다른 사람의 작업 내용을 확인하는 직접적인 방법을 제공합니다.
마지막으로 node와 npm의 설치가 필요합니다. 둘다 필요한 라이브러리를 관리하는데 쓰입니다. 이 책에서는 npm(node package manager, 노드 패키지 매니저)을 통해 외부 노드 패키지를 설치합니다. 노드 패키지는 라이브러리일 수도 있고 프레임워크 전체가 될 수도 있습니다.
터미널에서 node와 npm의 버전을 확인합시다. 터미널에 아무것도 출력되지 않으면 노드와 npm 부터 설치해주세요. 다음은 이 글을 작성하는 시점에 제가 사용한 버전들입니다.
{title="Command Line",lang="text"}
node --version
*v10.11.0
npm --version
*v6.9.0
이 부분은 노드와 npm에 대한 아주 짧은 설명입니다. 완벽하지는 않지만, 필요한 부분은 다룰 것입니다. 패키지의 설치 및 관리에 익숙하다면 이 부분을 건너뛰어도 좋습니다.
노드 패키지 매니저 (npm: node package manager)로 커맨드라인에서 외부 노드 패키지를 설치할 수 있습니다. 이들 패키지는 유틸리티 함수, 라이브러리 또는 프레임워크 전체가 될 수도 있는데, 애플리케이션은 설치된 패키지에 의존해서 동작하게 됩니다. 이때 패키지는 전역 노드 패키지 폴더 또는 프로젝트 내 지역 폴더에 설치될 수 있습니다.
전역 노드 패키지는 모든 터미널에서 액세스 할 수 있으며, 전역 폴더에 한 번만 설치하면 됩니다. 전역 패키지를 설치하는 명령은 다음과 같습니다.
{title="Command Line",lang="text"}
npm install -g <패키지 이름>
-g
옵션은 npm
에게 패키지의 전역 설치를 명령합니다. 지역 패키지는 현재 폴더에서 개발 중인 애플리케이션에서 사용됩니다. 우리의 목적에 따라 다음 명령으로 리액트를 지역 폴더에 설치합시다.
{title="Command Line",lang="text"}
npm install react
패키지는 생성된 node_modules/ 폴더에 저장되고 의존성 파일인 package.json에 패키지 리스트가 나열됩니다.
package.json 파일이 있어야만 npm
명령어로 node_modules/ 폴더를 초기화할 수 있습니다. 아래 명령어로 새 지역 패키지를 설치합니다.
{title="Command Line",lang="text"}
npm init -y
-y
옵션은 package.json을 기본값들로 초기화하라는 지시자입니다. 프로젝트를 초기화하고나면 npm install <package>
를 통해 새 패키지를 설치할 수 있습니다.
package.json을 이용하면 노드 패키지 파일 전부를 공유하지 않고도 다른 개발자와 프로젝트를 공유할 수 있습니다. 이 파일에는 프로젝트에 사용된 노드 패키지 목록이 전부 적혀 있는데, 이를 dependencies(애플리케이션이 의존하고 있는 패키지들을 부르는 명칭)라 부릅니다. 프로젝트를 복사할 때는 의존성 패키지들을 제외하고 복사해도 됩니다. package.json에 패키지 정보가 들어있으므로 프로젝트를 복사한 후 npm install
명령으로 모든 패키지를 쉽게 설치할 수 있게됩니다. npm install
스크립트는 package.json 파일에 나열된 모든 의존성 패키지 정보를 읽어서 node_modules/ 폴더에 설치해줍니다.
마지막으로 한가지 명령만 더 알아봅시다.
{title="Command Line",lang="text"}
npm install --save-dev <package>
--save-dev
옵션은 설치할 패키지가 개발 환경에서만 사용된다는 의미입니다. 개발한 애플리케이션을 서버에 배포하는 경우, 혹은 프로덕션 환경에서는 필요하지 않은 패키지를 설치할 때 사용합니다. 테스트에만 사용하고 프로덕션 환경에서는 제외할 패키지를 설치할 때 유용합니다.
어떤 분들은 노드 패키지 관리에 다른 프로그램을 사용하고 싶을 수도 있습니다. npm과 비슷한 의존성 관리자로 yarn이 있습니다. yarn은 명령어는 다르지만 npm 레지스트리에 접근할 수 있습니다. yarn은 npm의 문제점을 개선하기 위해 만들어졌습니다. 그러나 두 도구 모두 빠르게 발전했기 때문에 지금은 어느 것을 써도 상관없습니다.
- 테스트를 위한 npm 프로젝트를 터미널에서 생성합니다.
mkdir <folder_name>
를 입력해 새 폴더를 생성합니다.cd <folder_name>
생성한 폴더 내로 들어갑니다.npm init -y
또는npm init
을 실행합니다.npm install react
를 입력해 리액트를 지역 패키지로 설치합니다.- package.json 파일과 node_modules/ 폴더가 생성되었는지 확인합니다.
- react 노드 패키지를 제거하고 다시 설치하는 방법을 스스로 찾아봅니다.
- npm 공식 문서를 읽어봅니다.
- yarn 패키지 매니저에대해 읽어봅니다.
리액트 애플리케이션을 시작하는 방법에는 여러가지가 있습니다. 첫번째로 CDN(Content Delivery Network: 콘텐츠 전송 네트워크의 약자)을 이용하는 방법을 알아보겠습니다. CDN이 무엇인지는 지금 중요하지 않습니다. 우리는 이 책에서 CDN을 사용하지 않을 것입니다. 하지만 짧게 설명하는 정도는 괜찮겠지요. 많은 기업들이 고객에게 파일을 제공하기 위해 CDN을 사용합니다. 이런 파일중에는 리액트와 같은 라이브러리도 있습니다. 배포용으로 포장된 리액트 라이브러리는 react.js라는 자바스크립트 파일이기 때문입니다.
CDN으로 리액트를 사용하려면 HTML 파일 내에 <script>
태그로 CDN url을 가리키면 됩니다. react와 react-dom 두 라이브러리 url을 추가합니다.
{title="Code Playground",lang="javascript"}
<script
crossorigin
src="https://unpkg.com/react@16/umd/react.development.js"
></script>
<script
crossorigin
src="https://unpkg.com/react-dom@16/umd/react-dom.development.js"
></script>
노드 프로젝트로 초기화해서 리액트를 사용할 수도 있습니다. 애플리케이션 내 package.json 파일이 있다면, npm 명령어로 react와 react-dom을 설치할 수 있습니다. 제일 먼저 npm init -y 명령어를 입력해 package.json을 초기화합니다. 그 다음 npm 명령어 한 줄로 여러 노드 패키지를 한 번에 설치합니다.
{title="Command Line",lang="text"}
npm install react react-dom
이와 같은 방법은 현재 개발 중인 애플리케이션에 리액트를 추가할 때도 사용합니다.
JSX(리액트 문법)와 자바스크립트 ES6로 만든 애플리케이션은 바벨(babel)이 필요합니다. 모든 브라우저가 ES6 문법을 지원하지는 않기때문에, 바벨을 통해 ES6와 JSX를 바닐라 자바스크립트로 변환해야 합니다. 바벨은 설치가 어렵기 때문에 페이스북은 누구나 손쉽게 환경 설정을 할 수 있는 제로 구성 설치(zero-configuration) 솔루션인 create-react-app를 만들었습니다. 우리는 이 패키지로 리액트 애플리케이션을 만들어 볼 것입니다.
이 책에서는 create-react-app으로 애플리케이션 작성을 시작합니다. create-react-app은 2016년 페이스북이 발표한 설정없는 시작 도구(Zero-Configuration Setup Starter Kit)입니다. 어떤 조사에 따르면 96% 이상의 리액트 개발자들이 입문자에게 추천하는 도구라고 합니다. create-react-app의 개발자들이 안보이는 곳에서 도구와 설정을 계속 발전시키고 있기 때문에, 우리는 오롯이 애플리케이션 구현에만 신경 쓰면 됩니다.
전역 노드 패키지에 create-react-app 패키지를 설치해야 커맨드라인에서 리액트 애플리케이션 작성을 시작할 수 있습니다.
{title="Command Line",lang="text"}
npm install -g create-react-app
create-react-app 버전을 확인해, 성공적으로 패키지가 설치되었는지 확인합시다.
{title="Command Line",lang="text"}
create-react-app --version
*v2.0.2
이제 여러분의 첫 번째 리액트 애플리케이션 작성을 시작해보겠습니다. 이름은 hackernews라고 하겠습니다. 물론 다른 이름을 사용해도 괜찮습니다. 생성 후에는 폴더안에 들어가보겠습니다.
{title="Command Line",lang="text"}
create-react-app hackernews
cd hackernews
코드 에디터에서 애플리케이션을 열어 봅시다. create-react-app의 버전에 따라 조금 다를 수도 있지만, 아래와 같은 폴더 구조가 보일 것입니다.
{title="Folder Structure",lang="text"}
hackernews/
README.md
node_modules/
package.json
.gitignore
public/
favicon.ico
index.html
manifest.json
src/
App.css
App.js
App.test.js
index.css
index.js
logo.svg
serviceWorker.js
각 파일과 폴더에 대해 알아보겠습니다.
-
README.md: .md 확장자는 마크다운(markdown) 파일이라는 의미입니다. 일반 텍스트와 함께 간단한 마크업 언어로 작성합니다. 대부분의 프로젝트는 README.md 파일에 프로젝트 설명 및 설치 방법 등 안내 사항을 작성합니다. 깃허브의 프로젝트에서 README.md 파일의 내용이 표시된 것을 본 적이 있을 겁니다. create-react-app 을 설치한 후 바로 깃허브에 프로젝트를 올린다면 README.md 파일 내용은 create-react-app 공식 깃허브 리퍼지토리와 내용이 동일할 것입니다.
-
node_modules/: 이 폴더에는 npm으로 설치된 모든 노드 패키지가 들어있습니다. create-react-app 으로 리액트 애플리케이션의 뼈대를 만들었기 때문에 이미 여러 모듈이 설치되어 있습니다. 이 폴더는 절대로 건드리지 말아야 합니다. 패키지를 설치하거나 제거할 때는
npm
명령어를 사용하도록 합니다. -
package.json: 애플리케이션이 사용하는 노드 패키지 목록 및 기타 프로젝트 설정이 저장됩니다.
-
.gitignore: git 사용시, 원격 리퍼지토리에 공유하지 않을 모든 파일과 폴더가 나열되어 있습니다. 예를 들어 node_module은 로컬에만 있어야 합니다. 해당 폴더를 공유하지 않아도 package.json 파일만 공유하면
npm install
명령으로 필요한 패키지들을 설치할 수 있기 때문입니다. -
public/: 이 폴더에는 public/index.html 처럼 개발에 필요한 파일이 들어있습니다. index.html은 개발중에 localhost:3000 에 접속했을때 표시되는 파일입니다. 자동 생성된 파일들이 이 index.html과 src/ 폴더 안의 스크립트들을 알아서 연결해줍니다.
-
build/: 프로덕션 빌드 시 생성되는 폴더입니다. 빌드 시 *src/*와 public/ 폴더에 있는 파일들이 배포용으로 묶여서 build 폴더에 저장됩니다.
-
manifest.json 및 serviceWorker.js: 지금 단계에서는 무시해도 됩니다. 이 프로젝트에서는 사용하지 않은 파일입니다.
지금 단계에서 우리에게 필요한 파일들은 모두 src/ 폴더 안에 있습니다. 제일 중요한 파일은 src/App.js로써 리액트 컴포넌트 생성에 사용됩니다. 지금은 이 파일이 전체 애플리케이션을 이루고 있지만, 나중에 컴포넌트 별로 파일을 나눠 관리할 수 있습니다.
이외에 테스트를 위한 src/App.test.js와 리액트의 진입점인 src/index.js가 보일 겁니다. 다음 장에서 두 파일에 대해 설명합니다. 또한 애플리케이션과 컴포넌트의 스타일을 지정하는 src/index.css 및 src/App.css가 있습니다. 현재는 기본 스타일만 적용되어 있습니다.
create-react-app 애플리케이션은 npm 프로젝트로써 노드 패키지를 설치하고 제거할 수 있습니다. 그리고 커맨드라인에서 사용할 수 있는 아래와 같은 npm 스크립트를 제공합니다.
{title="Command Line",lang="text"}
// http://localhost:3000 에서 애플리케이션 실행
npm start
// 테스트 실행
npm test
// 배포를 위한 애플리케이션 빌드
npm run build
이 스크립트들은 package.json에 정의되어있습니다. 이것으로 리액트 애플리케이션 작성이 시작되었습니다. 이제 브라우저를 열어 리액트 애플리케이션이 구동되는지 확인해봅시다.
- 소스코드가 생성 후 상태와 동일한지 확인합니다.
npm start
을 실행해 브라우저에서 애플리케이션을 확인합니다. (끝내려면 Control+C를 누릅니다.)npm test
를 실행합니다.npm run build
스크립트를 실행하고 build/ 폴더가 추가되었는지 확인합니다. (폴더는 삭제해도 됩니다. 나중에 배포할 때 사용된다는 것만 기억하세요.)- 파일들의 내용에 익숙해집니다.
- npm 스크립트와 create-react-app를 읽어봅니다.
리액트의 JSX 문법에 대해 알아봅시다. create-react-app으로 애플리케이션의 뼈대를 생성하면 모든 파일들이 기본적인 동작을 수행할 수 있는 상태로 만들어집니다. 우리는 src/App.js 파일 만을 수정하도록 하겠습니다.
{title="src/App.js",lang=javascript}
import React, { Component } from 'react';
import logo from './logo.svg';
import './App.css';
class App extends Component {
render() {
return (
<div className="App">
<header className="App-header">
<img src={logo} className="App-logo" alt="logo" />
<p>
Edit <code>src/App.js</code> and save to reload.
</p>
<a
className="App-link"
href="https://reactjs.org"
target="_blank"
rel="noopener noreferrer"
>
Learn React
</a>
</header>
</div>
);
}
}
export default App;
import
과 export
혹은 클래스(class) 문법을 모른다고 걱정할 필요는 없습니다. 자바스크립트 ES6의 기능인데 잠시후에 설명하겠습니다.
App.js
파일에는 App이라는 이름의 **리액트 ES6 클래스 컴포넌트(React ES6 Class Component)**가 있습니다. 이렇게 컴포넌트를 선언하고 나면, 애플리케이션의 요소로써 어디서든 사용할 수 있습니다. 컴포넌트를 사용하면 컴포넌트의 인스턴스가 만들어지는데, 다른 말로는 컴포넌트의 인스턴스화 라고도 합니다.
{title="Code Playground",lang="javascript"}
// 컴포넌트 선언
class App extends Component {
...
}
// 컴포넌트 사용 (클래스의 인스턴스화라고도 한다.)
// 컴포넌트의 인스턴스를 생성한다.
<App />
render()
메서드는 **엘레먼트(element)**를 돌려줍니다. 앞서 인스턴스를 만들었던 컴포넌트는 여러 엘레먼트가 모여서 만듭니다. 따라서 컴포넌트(component), 컴포넌트의 인스턴스(instance), 엘레먼트(element) 간의 차이를 이해하는 것이 중요합니다.
App
컴포넌트가 인스턴스화 된 것을 볼 수 있는데, 만약 인스턴스화 되지 않았다면 브라우저에서 렌더링 된 결과를 볼 수 없었을 것입니다. App 컴포넌트는 선언일 뿐이고, 사용된 것은 아닙니다. JSX 문법으로 <App/>
라고 적으면 인스턴스화 하는 것입니다. 잠시후에 사용하는 장면을 보여드리겠습니다.
render()
블록 안 내용은 HTML과 비슷해 보이지만, JSX 문법으로 작성되었습니다. JSX 문법은 HTML과 자바스크립트를 섞어 쓸수 있게 해줍니다. 강력한 문법이지만, 지금까지 HTML과 자바스크립트를 따로 분리해서 사용해왔다면 꽤 혼란스러울 것입니다. 먼저 JSX에서 기본적인 HTML 엘레먼트를 작성해보겠습니다. App.js
의 모든 내용을 지우고 아래와 같이 바꿔봅시다.
{title="src/App.js",lang=javascript}
import React, { Component } from 'react';
import './App.css';
class App extends Component {
render() {
return (
<div className="App">
<h2>리액트 도움닫기에 오신 여러분을 환영합니다</h2>
</div>
);
}
}
export default App;
이 코드의 render()
메서드는 자바스크립트가 없는 HTML만을 반환했습니다. 이제 변수를 만들어 "리액트 도움닫기에 오신 여러분을 환영합니다"라는 값을 주고, 이 변수를 JSX 문법인 중괄호({}
) 안에 넣어봅시다.
{title="src/App.js",lang=javascript}
import React, { Component } from 'react';
import './App.css';
class App extends Component {
render() {
# leanpub-start-insert
var helloWorld = '리액트 도움닫기에 오신 여러분을 환영합니다';
# leanpub-end-insert
return (
<div className="App">
# leanpub-start-insert
<h2>{helloWorld}</h2>
# leanpub-end-insert
</div>
);
}
}
export default App;
다시 npm start
명령으로 애플리케이션을 실행해서 수정 사항이 반영되었는지 확인합니다.
아마 className
속성(attribute)이 눈에 띌 것 같습니다. className
은 표준 HTML의 class
로부터 왔습니다. JSX는 몇 가지 HTML 내부 속성을 바꿔야 했는데, 리액트의 HTML 속성 지원에 목록이 있습니다. 바뀐 이름들은 모두 카멜 케이스(camelCase) 표기법을 따릅니다. 앞으로 JSX 속성 몇개를 더 만나게 될 것입니다.
-
지난 섹션의 코드를 확인합니다.
- 이번 섹션에서 변경된 코드를 확인합니다.
-
JSX 안에 새 변수를 만들고 렌더링 합니다.
user
객체를 만들고 JSX 문법으로firstname
과lastname
을 표시합니다.render()
메서드 안에서user
프로퍼티를 사용합니다.
-
[리액트 공식 문서] JSX를 읽어봅니다.
-
[리액트 공식 문서] 리액트 컴포넌트, 엘레먼트, 인스턴스를 읽어봅니다.
이전 장에서 var
문을 사용해 변수 helloWorld
를 선언했습니다. 자바스크립트 ES6에서 변수 선언은 const
또는 let
을 사용합니다. ES6에서는 var
를 사용하는 일은 거의 없습니다.
const
로 선언된 변수는 다시 할당하거나 선언할 수 없습니다. 불변 데이터 구조(immutable data structures)이기 때문에 한 번 데이터 구조가 정의된 이후, 다시 변경할 수 없습니다.
{title="Code Playground",lang="javascript"}
// 불가능
const helloWorld = '리액트에 오신 여러분을 환영합니다';
helloWorld = '안녕 리액트';
반면 let
은 변경 가능합니다. 아래와 같이 변수를 다시 할당할 수 있습니다.
{title="Code Playground",lang="javascript"}
// 가능
let helloWorld = '리액트에 오신 여러분을 환영합니다';
helloWorld = '안녕 리액트';
기본적으로 const
로 선언된 변수는 변경할 수 없습니다. 그러나 변수가 배열이나 객체 타입일 경우 변경 가능합니다.
{title="Code Playground",lang="javascript"}
// 가능
const helloWorld = {
text: '리액트에 오신 여러분을 환영합니다'
};
helloWorld.text = '안녕 리액트';
그렇다면 언제 let
또는 const
를 사용해야 할까요? 정해진 법칙은 없지만, 가능하면 기본적으로 const
를 사용하는 것이 좋습니다. const
를 사용함은 객체와 배열의 값이 변경되더라도 데이터 구조를 변경되지 않음을 뜻하기 때문입니다. 반대로 변경되는 변수는 let
을 사용합니다.
리액트 생태계는 불변성 원칙을 준수하기 때문에, 변수를 정의할 때 let
보다 const
사용을 우선시 합니다. 복잡한 객체에서 내부 값을 수정해야 되는 경우도 있으니 const
사용에 주의합시다.
이 책은 var
대신 const
을 사용합니다. var
을 const
로 바꿉니다.
{title="src/App.js",lang=javascript}
import React, { Component } from 'react';
import './App.css';
class App extends Component {
render() {
# leanpub-start-insert
const helloWorld = '리액트에 오신 여러분을 환영합니다';
# leanpub-end-insert
return (
<div className="App">
<h2>{helloWorld}</h2>
</div>
);
}
}
export default App;
- 불변 데이터 구조에 대해 더 알아봅시다.
- 보편적인 프로그래밍에서 말하는 불변 데이터 구조란 무엇인지 알아봅시다.
- 리액트와 그 생태계에서 불변 데이터 구조가 사용되는 이유에 대해 알아봅시다.
App 컴포넌트는 리액트 세계로의 진입점인 src/index.js에서 사용됩니다.
{title="src/index.js",lang=javascript}
import React from 'react';
import ReactDOM from 'react-dom';
import App from './App';
import './index.css';
ReactDOM.render(
<App />,
document.getElementById('root')
);
ReactDOM.render()
메서드는 HTML의 DOM 노드를 JSX로 바꿔줍니다. 이 기능을 이용하면 기존 애플리케이션에도 리액트를 쉽게 통합할 수 있습니다. ReactDOM.render()
는 애플리케이션 내에서 여러 번 사용할 수 있습니다. JSX 구문이나 한 개 혹은 여러 개의 리액트 컴포넌트, 또는 전체 애플리케이션을 삽입할 때 사용 할 수 있습니다. 보통의 리액트 애플리케이션에서는 컴포넌트 트리를 불러오기 위해 한 번만 호출합니다.
ReactDOM.render()
에는 두 개의 인자가 필요합니다. 첫 번째 인자는 화면에 표시할 JSX, 두 번째 인자는 리액트 애플리케이션이 들어갈 HTML상의 위치입니다. 예제에서는 id
가 root
인 엘레먼트에 넣으려 하는데, public/index.html 파일에 해당 엘레먼트가 있습니다.
{title="Code Playground",lang=javascript}
ReactDOM.render(
<h1>안녕 리액트</h1>,
document.getElementById('root')
);
앞선 구현에서는 ReactDOM.render()
에 App 컴포넌트를 전달했지만, 위의 코드와 같이 간단한 JSX를 인자로 줄 수도 있습니다. 반드시 컴포넌트의 인스턴스를 주어야 하는 것은 아닙니다.
- public/index.html 파일을 열어 리액트 애플리케이션이 삽입될 HTML상의 위치를 찾아봅니다.
- 리액트에서 엘레먼트 렌더링하기 읽어보기
더 나은 개발 환경을 위해 src/index.js 파일에서 할 일이 하나 더 있습니다. 이는 필수가 아닌 옵션 사항으로 리액트 입문자가 꼭 해야 할 내용은 아닙니다.
create-react-app은 소스 코드 변경 시, 자동으로 브라우저를 새로고침 합니다. src/App.js 파일 내 helloWorld
를 다른 변수로 변경해보면 브라우저가 자동으로 새로고침 됩니다. 때문에 매번 수정할 때마다 새로고침을 하지 않아 편리합니다. 그러나 더 좋은 방법이 있습니다.
Hot Module Replacement(핫 모듈 리플레이스먼트, 이하 HMR)란 브라우저 내 애플리케이션을 재실행하는 도구입니다. create-react-app에서도 쉽게 사용할 수 있습니다. 이를 위해 src/index.js 파일을 열고 아래와 같이 설정을 추가하겠습니다.
{title="src/index.js",lang=javascript}
import React from 'react';
import ReactDOM from 'react-dom';
import App from './App';
import './index.css';
ReactDOM.render(
<App />,
document.getElementById('root')
);
# leanpub-start-insert
if (module.hot) {
module.hot.accept();
}
# leanpub-end-insert
위 코드가 전부입니다. src/App.js 파일에서 helloWorld
변수를 다른 것으로 바꿔봅시다. 브라우저는 페이지 새로고침 되지 않지만, 애플리케이션이 재실행되어 올바른 결과가 출력됩니다. 이처럼 HMR은 장점이 많습니다.
console.log()
로 디버깅을 해봅시다. 코드를 수정해도 브라우저 페이지가 새로고침 되지 않아 개발자 도구 콘솔에 과거 출력 히스토리가 그대로 남아있습니다. 디버깅이 훨씬 쉬워질 것입니다.
애플리케이션이 점차 고도화되면서 페이지 새로고침은 개발 생산성을 지연시킵니다. 페이지가 로드될 때까지 기다려야 하며, 대규모 애플리케이션에서 페이지를 다시 로드하는데 몇 초 이상이 걸릴 수 있습니다. HMR은 이러한 단점을 해결합니다.
가장 큰 장점은 HMR로 애플리케이션 상태를 유지할 수 있다는 것입니다. 애플리케이션에 3단계의 대화창이 있다고 가정해봅시다. 쉽게 다이얼로그 창 기능이라고 생각하면 됩니다. HMR이 없다면 소스 코드를 변경할 때마다 브라우저 페이지가 새로고침 됩니다. 매번 창을 다시 열어 1단계에서 3단계로 이동시켜야 하는 번거로운 일을 해야 합니다. HMR을 사용하면 다이얼로그 창은 3단계 상태로 유지되며, 소스 코드가 수정되어도 애플리케이션의 상태는 그대로 유지됩니다. 애플리케이션 자체는 다시 실행되지만 페이지는 새로고침 되지 않습니다.
- src/App.js 코드를 수정해 HMR이 어떻게 실행되는지 확인합니다.
- 댄 애브라몹(Dan Abramov)의 리액트 라이브 : Hot Reloading으로 시간 여행 떠나기 10분간 시청합니다.
App 컴포넌트로 다시 돌아갑시다. 지금까지 JSX에서 초기 변수를 렌더링 했습니다. 이제 배열의 각 항목을 차례로 렌더링 해봅시다. 지금은 샘플 데이터를 사용하지만 앞으로 외부 API를 통해 데이터를 가져올 것입니다. 앞으로 점점 재밌고 흥미 있어질 거니 기대해도 좋습니다.
먼저 아래와 같이 list
를 정의합시다.
{title="src/App.js",lang=javascript}
import React, { Component } from 'react';
import './App.css';
# leanpub-start-insert
const list = [
{
title: 'React',
url: 'https://reactjs.org/',
author: 'Jordan Walke',
num_comments: 3,
points: 4,
objectID: 0,
},
{
title: 'Redux',
url: 'https://github.com/reactjs/redux',
author: 'Dan Abramov, Andrew Clark',
num_comments: 2,
points: 5,
objectID: 1,
},
];
# leanpub-end-insert
class App extends Component {
...
}
list
는 객체로 이루어진 배열입니다. 각 프로퍼티에는 제목(title), url, 작성자(author), 식별자(objectID), 기사의 인기도를 나타내는 점수(point), 댓글 수(num_comments)가 있습니다.
이제 JSX에서 자바스크립트 반복 메서드인 map()
을 사용해 배열의 모든 데이터를 출력해봅시다. JSX는 자바스크립트 표현식을 캡슐화하기 위해 중괄호({}
)를 사용합니다.
{title="src/App.js",lang=javascript}
class App extends Component {
render() {
return (
<div className="App">
# leanpub-start-insert
{list.map(function(item) {
return <div>{item.title}</div>;
})}
# leanpub-end-insert
</div>
);
}
}
export default App;
JSX는 HTML과 자바스크립트를 함께 사용합니다. map()
를 사용해 HTML 태그와 섞어 엘레먼트를 출력해보겠습니다.
현재는 title
프로퍼티 값만 출력됩니다. 이제 모든 프로퍼티 값을 보여줍시다.
{title="src/App.js",lang=javascript}
class App extends Component {
render() {
return (
<div className="App">
# leanpub-start-insert
{list.map(function(item) {
return (
<div>
<span>
<a href={item.url}>{item.title}</a>
</span>
<span>{item.author}</span>
<span>{item.num_comments}</span>
<span>{item.points}</span>
</div>
);
})}
# leanpub-end-insert
</div>
);
}
}
export default App;
JSX와 map()
메서드로 반복문을 작성했습니다. 각 프로퍼티를 HTML <span>
태그로 감싸고, url은 a 태그의 href 속성과 함께 사용했습니다.
그러나 아직 한 가지가 남았습니다. 바로 각 엘레먼트마다 key
속성을 추가해야 합니다. 리액트는 key
로 배열의 수정 및 제거된 항목을 식별합니다. 우리는 objectID
프로퍼티를 key
로 지정하겠습니다.
{title="src/App.js",lang=javascript}
{list.map(function(item) {
return (
# leanpub-start-insert
<div key={item.objectID}>
# leanpub-end-insert
<span>
<a href={item.url}>{item.title}</a>
</span>
<span>{item.author}</span>
<span>{item.num_comments}</span>
<span>{item.points}</span>
</div>
);
})}
key
값은 고윳값으로 식별 가능해야 합니다. 배열의 인덱스 값은 고정값이 아니기 때문에 사용하지 말아야 합니다. 배열의 순서가 변경되면, 리액트가 각 항목을 식별할 수 없기 때문입니다.
{title="src/App.js",lang=javascript}
// 이렇게 사용하면 안됩니다.
{list.map(function(item, key) {
return (
<div key={key}>
...
</div>
);
})}
이제 두 배열의 모든 항목이 보입니다. 브라우저를 열어 변경 내용을 확인합시다.
- JSX 내 자바스크립트 표현식(expression)을 작성합니다.
자바스크립트 ES6에는 새로운 문법인 화살표 함수(Arrow function)가 도입되었습니다. 화살표 함수를 사용하면 기존 함수 표현식보다 코드가 간결해집니다.
{title="Code Playground",lang="javascript"}
// 함수 표현식
function () { ... }
// 화살표 함수 표현식
() => { ... }
만일 매개변수가 하나라면 괄호 생략이 가능하지만, 복수 매개변수이거나 매개변수가 없는 함수는 괄호를 생략할 수 없습니다.
{title="Code Playground",lang="javascript"}
// 가능
item => { ... }
// 가능
(item) => { ... }
// 불가능
item, key => { ... }
// 가능
(item, key) => { ... }
앞서 작성한 map()
부분을 ES6 화살표 표현식으로 고쳐봅시다.
{title="src/App.js",lang=javascript}
# leanpub-start-insert
{list.map(item => {
# leanpub-end-insert
return (
<div key={item.objectID}>
<span>
<a href={item.url}>{item.title}</a>
</span>
<span>{item.author}</span>
<span>{item.num_comments}</span>
<span>{item.points}</span>
</div>
);
})}
ES6 화살표 함수에서는 중괄호({}
) 부분인 *블록 본문(block body)*을 생략할 수 있습니다. 이렇게 *간결한 본문(concise body)*를 사용하면 자동으로 return 명령이 추가되는 효과가 있습니다. 따라서 return
문을 삭제할 수 있습니다. 화살표 함수를 자주 보게 될 것이기 때문에, 화살표 기능을 사용할 때 블록 본문과 간결한 본문의 차이점을 이해하고 있어야 합니다.
{title="src/App.js",lang=javascript}
# leanpub-start-insert
{list.map(item =>
# leanpub-end-insert
<div key={item.objectID}>
<span>
<a href={item.url}>{item.title}</a>
</span>
<span>{item.author}</span>
<span>{item.num_comments}</span>
<span>{item.points}</span>
</div>
# leanpub-start-insert
)}
# leanpub-end-insert
함수 선언, 중괄호, return 문이 생략되면서 JSX 코드가 더 간결하고 보기 좋아졌습니다.
-
- 이 섹션에서 수정된 부분 확인하기
-
ES6 화살표 함수 읽어보기
자바스크립트 ES6에서 클래스(Class)가 도입되었습니다. 클래스는 일반적으로 객체 지향 프로그래밍 언어에서 사용합니다. 자바스크립트는 융통성과 유연성이 뛰어난 프로그래밍 언어로 함수형 프로그래밍과 객체지향 프로그래밍으로 모두 작성할 수 있습니다.
리액트는 함수형 프로그래밍과 객체 프로그래밍의 모든 장점을 취하고 있습니다. 리액트는 불변하는 데이터 구조를 다룰 때는 함수형 프로그래밍을, 컴포넌트를 만들 때는 클래스로 선언합니다.
다음 Developer
클래스를 리액트 컴포넌트가 아닌, ES6 클래스로 간주해봅시다.
{title="Code Playground",lang="javascript"}
class Developer {
constructor(firstname, lastname) {
this.firstname = firstname;
this.lastname = lastname;
}
getName() {
return this.firstname + ' ' + this.lastname;
}
}
클래스에는 인스턴스화 할 수 있는 생성자(constructor)가 있습니다. 생성자는 매개변수를 사용해 클래스 인스턴스에 할당합니다. 또한 클래스는 함수를 정의합니다. 함수는 클래스와 연관되어 있어 메서드(method)라고도 불립니다. 이 메서드는 클래스 메서드로 참조됩니다.
Developer 클래스는 클래스 선언문일 뿐입니다. 클래스를 호출해 클래스 내 여러 인스턴스를 작성할 수 있습니다. ES6 클래스 컴포넌트와 비슷하지만, 다른 인스턴스를 사용하여 인스턴스를 생성해야 합니다.
클래스를 인스턴스 화하는 방법과 클래스를 사용하는 방법을 살펴봅시다.
{title="Code Playground",lang="javascript"}
const robin = new Developer('Robin', 'Wieruch');
console.log(robin.getName());
// 출력: Robin Wieruch
리액트는 ES6 클래스 컴포넌트에 자바스크립트 ES6 클래스를 사용합니다. 이미 하나의 ES6 클래스 컴포넌트를 사용했습니다.
{title="src/App.js",lang=javascript}
import React, { Component } from 'react';
...
class App extends Component {
render() {
...
}
}
App 클래스는 Component
로 확장(extend)됩니다. 기본적으로 App 컴포넌트에서 확장을 선언했지만, 다른 컴포넌트에서도 선언할 수 있습니다. 그렇다면 확장(extend)이란 무엇일까요? 객체 지향 프로그래밍에는 상속의 원칙이 있습니다. 한 클래스의 기능을 다른 클래스로 전달하는 데 사용됩니다.
App 클래스는 Component
클래스의 기능을 확장함으로, Component
기능을 상속받습니다. Component
클래스는 ES6 클래스를 ES6 컴포넌트 클래스로 확장됩니다. 따라서 컴포넌트 클래스 메서드를 쓸 수 있는 것입니다. 그중 render()
은 우리가 이미 사용했던 컴포넌트 클래스 메서드입니다. 앞으로 여러 컴포넌트 클래스 메서드에 대해 배울 것입니다.
Component
클래스는 리액트 컴포넌트의 모든 구현 세부 사항을 캡슐화하기 때문에 리액트에서 클래스를 컴포넌트로 사용할 수 있습니다.
지금까지 ES6 클래스 기초와 이를 리액트 컴포넌트로 사용하는 방법을 배웠습니다. 컴포넌트 메서드는 리액트 생명주기(Life Cycle) 절에서 자세히 배워볼 것입니다.
{pagebreak}
1장에서는 create-react-app으로 리액트 애플리케이션을 부트스트래핑을 했습니다. 지금까지 학습한 내용을 정리해봅시다.
-
리액트
- create-react-app으로 리액트 애플리케이션을 부트스트래핑 합니다.
- JSX 문법으로
render()
메서드 내 HTML과 자바스크립트를 같이 사용해 컴포넌트의 결괏값을 출력합니다. - 컴포넌트, 인스턴스, 엘레먼트는 서로 다릅니다.
ReactDOM.render()
는 리액트 애플리케이션이 리액트를 DOM에 연결시키는 엔트리 포인트(entry point)입니다.- JSX에서 자바스크립트 내장 함수를 사용할 수 있습니다.
- 배열의 엘레먼트를 렌더링 할 때
map()
와 HTML 구문과 함께 사용합니다.
-
ES6
- 상황에 따라
const
와let
로 변수를 선언합니다. - 리액트에서는 가능한
const
로 변수를 선언합니다. - 화살표 함수를 사용해 간결한 함수로 만듭니다.
- 클래스를 확장하여 리액트 컴포넌트로 정의합니다.
- 상황에 따라
잠시 휴식 시간을 가집시다. 학습 내용을 되새기고 적용해보며 지금까지 작성한 코드와 배운 내용을 내 것으로 만듭시다. 이것저것 만들어보며 테스트해보길 바랍니다.
앞으로 만들어볼 해커 뉴스 애플리케이션이 궁금하다면 깃허브 리퍼지토리를 확인하세요.