728x90
SMALL

목차

1. 화면을 그린 후 지우고 다시 출력하면 깜빡이는 문제점

2. 게임 프로그래밍에서의 버퍼 개념

3. 페이지 전환을 통해 깜빡이는 문제 해결하기


1. 화면을 그린 후 지우고 다시 출력하면 깜빡이는 문제점

C언어게임은 콘솔창이므로 콘솔창에서는 고속으로 화면을 지우고 그리게 하는 부분이 없어 지우고 다시 그리는 과정에서 깜빡임이 발생하게 된다.

 

아래 구조는 간단하게 구조체를 하나 만들고 방향키로 캐릭터가 움직이도록 구현하였다.

#include <stdio.h>
#include<Windows.h>
void gotoxy(int x, int y)   //gotoxy api함수

{
	COORD pos = { x,y };
	SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE), pos);
}

typedef struct _PLAYER
{
	int x, y; 
}PLAYER;

PLAYER player;

void init() {
	player.x = 20;
	player.y = 10;
}
void update() {
}
void render()
{
	system("cls");
	gotoxy(player.x, player.y);
	printf("♀");
}

void release() {
}

int main(void) {
	int nKey;
    
	init();

	while (1) {
		if (_kbhit()) { //키보드가 눌렸는지 체크 
			nKey = _getch(); //눌린값 대입
			if (nKey == 'q') //q를 눌렀다면 break
				break;
			else if (nKey == 224 || nKey == 0) {
				nKey = _getch();
			switch (nKey) {
			case 75: //왼쪽 방향키를 눌렀다면 
				player.x--; 
				break;
			case 77:  //오른쪽 방향키를 눌렀다면
				player.x++; 
				break;
			
			case 72:  //위쪽 방향키를 눌렀다면
				player.y--; 
				break;
			
			case 80:  //아래쪽 방향키를 눌렀다면
				player.y++; 
				break;
		}
			}
		}
		update();
		render();
	}
	release();
	return 0;
}

이동하기 전 위치의 문자를 보이지 않게 하기 위해 system("cls")로 지우고 다시 새로운 위치에 문자를 출력시키기 때문에

어쩔수없이 화면이 지워졌다가 다시 그려짐으로 인해서 화면이 깜빡이게 된다.

 

그렇다고 system("cls")를 해주지 않으면 다음과 같이 잔상이 남게 된다

 

 

이런 화면깜빡임은 보는 사람에게 너무 피로감을 느끼게 한다.  이런 문제를 해결하는 방법을 알아보자.


2. 게임 프로그래밍에서의 버퍼 개념

먼저 해결하는 방법을 알기 위해서는 게임 프로그래밍에서 쓰는 버퍼 개념을 알아야한다.

 

1) 전위 버퍼(primary buffer)

전위 버퍼를 전위면이라고도 하는데 화면과 일대일 대응되는 메모리르 ㄹ말하며 그래픽 카드의 메인 메모리의 일부분이다.

현재 모니터의 해상도를 변경하면 전위 버퍼에 해당되는 메모리도 변경된다.

 

2) 후위 버퍼(back buffer)

전위 버퍼와 동일한 특성을 가진 메모리를 말한다.

전위 버퍼는 모니터와 일대일 대응되므로 컴퓨터와 동작함과 동시에 그래픽 카드에 생성이 되지만 후위 법퍼는 따로 생성이 된다.

 

후위 버퍼는 전위 버퍼와 특성이 같으므로 전위 버퍼와 연결하여 사용하며 주로 시스템 메모리 보다는 그래픽 카드 메모리에 생성하여 빠른 화면 전환을 하기 위해 사용된다.

 

3) 이중 버퍼링(double buffering)

이중 버퍼링은 두 개의 버퍼를 이용하여 화면을 전환하는 방법을 말한다.

사용되는 두 개의 버퍼는 전위 버퍼와 후위 버퍼를 말하며 후위 버퍼의 내용을 전위 버퍼에 복사하여 출력하는 것을 말한다.

 

4) 페이지 전환(Page flipping)

이중 버퍼링에도 문제가 있어 오늘날 모든 게임에서 사용되는 방식이 바로 페이지 전환 방식이다.

실제로 전위 버퍼와 후위 버퍼간의 메모리 복사가 아닌 화면과 일대일 대응하는 메모리의 시작주소를 바꾸는 방식이다.

 

즉, 전위 버퍼를 쓰고 있다가 메모리의 시작 주소를 후위 버퍼로 바꿔서 후위 버퍼를 보여줬다가 다시 시작 주소를 바꿔서 전위 버퍼를 보여줬다가 번갈아가며 출력하게 하는 것이다.


3. 페이지 전환을 통해 깜빡이는 문제 해결하기

위에 봤던 간단한 예제를 버퍼 예제로 바꿔보자.

 

일단 기본 구조는 다음과 같다.

잘 모르겠다면 아래 링크를 잠깐 갔다 오자.

C언어로 게임만들기 1탄 - 게임의 기본구조 :: 잡코딩 정보 블로그 (tistory.com)

 

일단 코드를 복붙하고 따라해보자.

총 3개의 파일이 필요하다

1) main.c

#include <stdio.h>
#include<Windows.h>
#include "Screen.h"

typedef struct _PLAYER
{
	int x, y; //출력기준좌표
}PLAYER;

PLAYER player;

void init() {
	player.x = 20;
	player.y = 10;
}
void update() {
}
void render()
{
	   ScreenClear(); //대기하고 있는 화면 버퍼를 지운다.
	  ScreenPrint(player.x, player.y, "♀"); //(player.x, player.y) 좌표에 문자를 출력한다.
		ScreenFlipping(); //활성화된 화면 버퍼와 비활성화된 화면 버퍼의 상태를 바꾼다
}

void release() {
}

int main(void) {
	int nKey, nRemain;

	ScreenInit();//버퍼를 2개 생성한다.
	init();

	while (1) {
		if (_kbhit()) { //키보드가 눌렸는지 체크 
			nKey = _getch(); //눌린값 대입
			if (nKey == 'q') //q를 눌렀다면 break
				break;
			else if (nKey == 224 || nKey == 0) {
				nKey = _getch();
			switch (nKey) {
			case 75: //왼쪽 방향키를 눌렀다면 
				player.x--; 
				break;
			case 77:  //오른쪽 방향키를 눌렀다면
				player.x++; 
				break;
			
			case 72:  //위쪽 방향키를 눌렀다면
				player.y--; 
				break;
			
			case 80:  //아래쪽 방향키를 눌렀다면
				player.y++; 
				break;
		}
			}
		}
		update();
		render();
	}


	release();
	ScreenRelease(); //화면 버퍼 초기화 함수에서 생성한 두 개의 화면 버퍼를 모두 해제한다.
	return 0;
}

 

2) Screen.c (작명은 아무렇게나 해도 상관없다)

#include <stdio.h>
#include<Windows.h>

static int g_nScreenIndex;
static HANDLE g_hScreen[2];

void ScreenInit()
{
    CONSOLE_CURSOR_INFO cci;

    //화면 버퍼 2개를 만든다.
    g_hScreen[0] = CreateConsoleScreenBuffer(GENERIC_READ | GENERIC_WRITE,
        0, NULL, CONSOLE_TEXTMODE_BUFFER, NULL);
    g_hScreen[1] = CreateConsoleScreenBuffer(GENERIC_READ | GENERIC_WRITE,
        0, NULL, CONSOLE_TEXTMODE_BUFFER, NULL);

    //커서 숨기기
    cci.dwSize = 1;
    cci.bVisible = FALSE;
    SetConsoleCursorInfo(g_hScreen[0], &cci);
    SetConsoleCursorInfo(g_hScreen[1], &cci);
}

void ScreenFlipping()
{
    SetConsoleActiveScreenBuffer(g_hScreen[g_nScreenIndex]);
    g_nScreenIndex = !g_nScreenIndex;
}

void ScreenClear()
{
    COORD Coor = { 0,0 };
    DWORD dw;
    FillConsoleOutputCharacter(g_hScreen[g_nScreenIndex], ' ', 80 * 25, Coor, &dw);
}

void ScreenRelease()
{
    CloseHandle(g_hScreen[0]);
    CloseHandle(g_hScreen[1]);
}

void ScreenPrint(int x, int y, char* string)
{
    DWORD dw;
    COORD CursorPosition = { x, y };
    SetConsoleCursorPosition(g_hScreen[g_nScreenIndex], CursorPosition);
    WriteFile(g_hScreen[g_nScreenIndex], string, strlen(string), &dw, NULL);

}

 

3) Screen.h (작명은 아무렇게나 해도 된다)

void ScreenInit();
void ScreenFlipping();
void ScreenClear();
void ScreenRelease();
void ScreenPrint(int x, int y, char* string);
void SetColor(unsigned short color);

 

위에 깜빡이는 예제와 거의 똑같은데 다른 것은 윈도우 API함수가 들어간 것이다.

파일이 준비돼었으니 차근차근 보도록하자.

 

Screen.c 코드는 이해할필요없다. 그냥 가져다 쓰면 된다.

 

<1> ScreenInit()로 전위 버퍼와 백 버퍼로 사용할 화면 버퍼를 두 개 생성한다.

 

<2> ScreenClear()로 하나의 화면 버퍼가 활성화되어 출력되고 있는 동안에 다음 장면을 위한 화면 버퍼는 지운다.

 

 

<3>  ScreenPrint()로 대기 화면 버퍼의 x, y 좌표에 문자열을 출력해준다.

전위버퍼/후위버퍼

 

<4> ScreenFlipping()로 활성화된 화면 버퍼와 비활성화된 화면 버퍼의 상태를 바꾼다.

전위버퍼/후위버퍼

 

아주 빠른 속도로 일어나서 그냥 이동하는 것 처럼 보이지만 이런식으로 빠르게 바꿔치기 돼서 화면에 보이게 된다.

이렇게 코드를 짜면 아래와 같이 깜빡이지 않는 편안함을 느낄 수 있다


참고문헌

C를 이용한 게임프로그래밍-이태성

 

 

 

728x90
LIST
728x90
SMALL

목차

1. 게임의 기본구조


1. 게임의 기본구조

게임의 기본 구조는 크게 초기화, 데이터 갱신, 화면 출력, 해제로 나뉜다.

모든 게임이 이 구조로 돼있지는 않지만 대부분의 게임 구조가 이 구조로 되어있다.

사진 출저 : 아래 참조

1) 초기화

실제로 게임이 실행되기 이전에 게임에 필요한 기본 데이터를 읽고 각종 변수를 초기화한다.

초기화의 내용에는 캐릭터 초기화, 사운드 초기화, 패턴 도는 AI 초기화, 메모리 할당 등이 있다.

 

2) 데이터 갱신

말그대로 데이터가 갱신되는 것이다.

입력장치(주로 키보드나 마우스)에 의해 데이터가 갱신되거나

충돌등의 다양한 방식으로 데이터가 스스로 갱신된다.

 

3) 화면 출력

말그대로 출력하는 것이다.

 

4) 해제

동적으로 메모리를  해제하는 경우가 대부분이지만 초기화에서 생성한 객체를 해제하는 역할도 한다.

객체는 사운드 엔진, 물리 엔진, 그래픽 엔진등을 말한다.

 

 

다음은 게임 기본 구조에 사용할 함수들이며 주로 게임 엔진에 사용되는 함수명이다.

 

사진 출저 : 아래 참조

게임 프로그래밍을 하기 전에 이와 같은 구조를 만드는 것은 분업을 해야하기 때문이다.

사진 출처 : 아래참조

기본구조를 코드로 구현하면 다음과 같다.

 

4개의 함수가 각각 한번씩 실행되고 종료된다. 

#include <stdio.h>

void init() {

}

void update() {

}

void render() {

}

void release() {

}

int main(void) {
	init(); 
	update();
	render();
	release();
	return 0;
}

 

하지만 게임에서 데이터 갱신과 화면 출력은 계속돼야하므로 무한 반복하는 형태로 바꿔준다.

#include <stdio.h>

void init() {

}

void update() {

}

void render() {

}

void release() {

}

int main(void) {
	init();  //초기화
    
	while(1){
	update(); //데이터 갱신
	render(); //화면출력
	}
    
	release(); //해제
	return 0;
}

출처

1. C를 이용한 게임프로그래밍-이태성

2. 사진출처 :  3장. 게임의 기본 구조 (1/2) - C 게임 프로그래밍 - 나우캠퍼스 (daum.net)

 

728x90
LIST
728x90
SMALL

*이 게시글은 Do it! 클론 코딩 영화 평점 웹서비스를 보고 정리한 글입니다.


목차

1. HTML구조 만들어주고 장르 추가하기

2. CSS로 스타일링하기

3. 영화앱 여러 기능 추가하기-react-router-dom 설치하고 프로젝트 폴더 정리하기

4.  영화앱 여러 기능 추가하기-라우터 만들기

5. 영화앱 여러 기능 추가하기-내비게이션 만들어보기

6. 영화앱 여러 기능 추가하기-영화 상세 정보 기능 만들어보기

7. 영화앱 여러 기능 추가하기-리다이렉트 기능 만들어 보기

8. 영화 앱 깃허브에 배포하기


1. HTML 구조 만들어주고 장르 추가하기

그 다음 기본적인 스타일링을 해보도록 하자.

스타일링의 핵심은 CSS지만 CSS를 적용할 HTML도 필요하므로 HTML구조를 넣어준다.

 

HTML에서 class속성을 사용하려면 'class'를 그대로 쓰지만 리액트에서는 class 속성을 사용하기 위해서는 'class'가 아니라 'className'을 써야한다.

리액트는 JSX를 HTML로 변환하면서 className을 class로 다시 바꿔준다.

 

App.js

import React from 'react';
import axios from 'axios';
import Movie from './Movie';


class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };
  getMovies = async () => {
    const {
      data: {
        data: { movies },
      },
    } = await axios.get('https://yts-proxy.now.sh/list_movies.json?sort_by=rating');
    this.setState({ movies, isLoading: false });
  };
  componentDidMount() {
    this.getMovies();
  }
  render() {
    const { isLoading, movies } = this.state;
    return (
      <section className="container">
        {isLoading ? (
          <div className="loader">
            <span className="loader__text">Loading...</span>
          </div>
        ) : (
          <div className="movies">
            {movies.map((movie) => {
              return (
                <Movie
                  key={movie.id}
                  id={movie.id}
                  year={movie.year}
                  title={movie.title}
                  summary={movie.summary}
                  poster={movie.medium_cover_image}
                />
              );
            })}
          </div>
        )}
      </section>
    );
  }
}

export default App;

 

Movie.js

코드를 완성해보니 id props가 필요없어서 지운다.

import React from 'react';
import PropTypes from 'prop-types';

function Movie({ title, year, summary, poster }) {
  return (
    <div className="movie">
      <img src={poster} alt={title} title={title} />
      <div className="movie__data">
        <h3 className="movie__title">{title}</h3>
        <h5 className="movie__year">{year}</h5>
        <p className="movie__summary">{summary}</p>
      </div>
    </div>
  );
}

Movie.propTypes = {

  year: PropTypes.number.isRequired,
  title: PropTypes.string.isRequired,
  summary: PropTypes.string.isRequired,
  poster: PropTypes.string.isRequired,
};

export default Movie;

 

여기까지 만들고보니 영화에서 나름 중요한 장르 데이터가 안들어가있다.

추가해준다.

 

arrayOf(PropTypes.string)은 문자열을 원소로 하는 배열을 의미한다.

import React from 'react';
import PropTypes from 'prop-types';


function Movie({ title, year, summary, poster, genres }) {
  return (
    <div className="movie">
      <img src={poster} alt={title} title={title} />
      <div className="movie__data">
        <h3 className="movie__title">{title}</h3>
        <h5 className="movie__year">{year}</h5>
        <p className="movie__summary">{summary}</p>
      </div>
    </div>
  );
}

Movie.propTypes = {
  year: PropTypes.number.isRequired,
  title: PropTypes.string.isRequired,
  summary: PropTypes.string.isRequired,
  poster: PropTypes.string.isRequired,
  genres:PropTypes.arrayOf(PropTypes.string).isRequired,
};

export default Movie;

아래 App 컴포넌트에는

genres={movie.genres}를 추가한다.

import React from 'react';
import axios from 'axios';
import Movie from './Movie';


class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };
  getMovies = async () => {
    const {
      data: {
        data: { movies },
      },
    } = await axios.get('https://yts-proxy.now.sh/list_movies.json?sort_by=rating');
    this.setState({ movies, isLoading: false });
  };
  componentDidMount() {
    this.getMovies();
  }
  render() {
    const { isLoading, movies } = this.state;
    return (
      <section className="container">
        {isLoading ? (
          <div className="loader">
            <span classNAme="loader__text">Loading...</span>
          </div>
        ) : (
          <div className="movies">
            {movies.map((movie) => {
              return (
                <Movie
                  key={movie.id}
                  year={movie.year}
                  title={movie.title}
                  summary={movie.summary}
                  poster={movie.medium_cover_image}
                  genres={movie.genres}
                />
              );
            })}
          </div>
        )}
      </section>
    );
  }
}

export default App;

그리고 Movie 컴포넌트에 장르가 출력될 수 있도록 코드를 수정한다

<ul className="movie__genres">
            {genres.map((genre)=>{
                return <li className="movie__genre">{genre}</li>;

            })}
        </ul>
import React from 'react';
import PropTypes from 'prop-types';


function Movie({ title, year, summary, poster, genres }) {
  return (
    <div className="movie">
      <img src={poster} alt={title} title={title} />
      <div className="movie__data">
        <h3 className="movie__title">{title}</h3>
        <h5 className="movie__year">{year}</h5>
        <ul className="movie__genres">
            {genres.map((genre)=>{
                return <li className="movie__genre">{genre}</li>;

            })}
        </ul>
        <p className="movie__summary">{summary}</p>
      </div>
    </div>
  );
}

Movie.propTypes = {
  year: PropTypes.number.isRequired,
  title: PropTypes.string.isRequired,
  summary: PropTypes.string.isRequired,
  poster: PropTypes.string.isRequired,
  genres:PropTypes.arrayOf(PropTypes.string).isRequired,
};

export default Movie;

 

위와 같은 코드를 짜면 오류가 발생된다.

앞에서 잠깐 얘기했지만 컴포넌트를 여러개 출력할 때는 유일한 값을 이용하여 key props를 추가해야한다.

그런데 API에서 movies는 id라는 식별가능한 유일한 키가 있는데 genres에는 그런게 없다.

이런 경우에는 어떻게 해야할까?

map함수의 두번째 인자로 index를 주면된다.

자세한 사항은 map함수에 대해 찾아보기 바란다.

 

<ul className="movie__genres">
            {genres.map((genre,index)=>{
                return <li key={index}className="movie__genre">{genre}</li>;

            })}
        </ul>
import React from 'react';
import PropTypes from 'prop-types';


function Movie({ title, year, summary, poster, genres }) {
  return (
    <div className="movie">
      <img src={poster} alt={title} title={title} />
      <div className="movie__data">
        <h3 className="movie__title">{title}</h3>
        <h5 className="movie__year">{year}</h5>
        <ul className="movie__genres">
            {genres.map((genre,index)=>{
                return <li key={index} className="movie__genre">{genre}</li>;

            })}
        </ul>
        <p className="movie__summary">{summary}</p>
      </div>
    </div>
  );
}

Movie.propTypes = {
  year: PropTypes.number.isRequired,
  title: PropTypes.string.isRequired,
  summary: PropTypes.string.isRequired,
  poster: PropTypes.string.isRequired,
  genres:PropTypes.arrayOf(PropTypes.string).isRequired,
};

export default Movie;

2. CSS로 스타일링하기

우선 Movies.css와 App.css를 만든다.

그 후 Movie.js와 App.js에 import 시킨다.

 

그 다음 아래 코드를 그냥 복사한다.

css수업이 아니므로 자세한 사항은 생략한다.

 

App.css

* {
  box-sizing: border-box;
}

body {
  margin: 0;
  padding: 0;
  font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell,
    'Open Sans', 'Helvetica Neue', sans-serif;
  background-color: #eff3f7;
  height: 100%;
}

 

Movie.css

.movies .movie {
    background-color: white;
    margin-bottom: 70px;
    font-weight: 300;
    padding: 20px;
    border-radius: 5px;
    color: #adaeb9;
    box-shadow: 0 13px 27px -5px rgba(50, 50, 93, 0.25), 0 8px 16px -8px rgba(0, 0, 0, 0.3),
      0 -6px 16px -6px rgba(0, 0, 0, 0.025);
  }
  
  .movies .movie a {
    display: grid;
    grid-template-columns: minmax(150px, 1fr) 2fr;
    grid-gap: 20px;
    text-decoration: none;
    color: inherit;
  }
  
  .movie img {
    position: relative;
    top: -50px;
    max-width: 150px;
    width: 100%;
    margin-right: 30px;
    box-shadow: 0 30px 60px -12px rgba(50, 50, 93, 0.25), 0 18px 36px -18px rgba(0, 0, 0, 0.3),
      0 -12px 36px -8px rgba(0, 0, 0, 0.025);
  }
  
  .movie .movie__title,
  .movie .movie__year {
    margin: 0;
    font-weight: 300;
  }
  
  .movie .movie__title {
    margin-bottom: 5px;
    font-size: 24px;
    color: #2c2c2c;
  }
  
  .movie .movie__genres {
    list-style: none;
    padding: 0;
    margin: 0;
    display: flex;
    flex-wrap: wrap;
    margin: 5px 0px;
  }
  
  .movie__genres li,
  .movie .movie__year {
    margin-right: 10px;
    font-size: 14px;
  }

 

영화 요약을 180자로 제한하기 위해 아래 코드를 수정한다.

<p className="movie__summary">{summary.slice(0,180)}</p>
import React from 'react';
import PropTypes from 'prop-types';
import './Movie.css';

function Movie({ title, year, summary, poster, genres }) {
  return (
    <div className="movie">
      <img src={poster} alt={title} title={title} />
      <div className="movie__data">
        <h3 className="movie__title">{title}</h3>
        <h5 className="movie__year">{year}</h5>
        <ul className="movie__genres">
            {genres.map((genre,index)=>{
                return <li key={index} className="movie__genre">{genre}</li>;

            })}
        </ul>
        <p className="movie__summary">{summary.slice(0,180)}</p>
      </div>
    </div>
  );
}

Movie.propTypes = {
  year: PropTypes.number.isRequired,
  title: PropTypes.string.isRequired,
  summary: PropTypes.string.isRequired,
  poster: PropTypes.string.isRequired,
  genres:PropTypes.arrayOf(PropTypes.string).isRequired,
};

export default Movie;

 

뜨는 제목을 바꾸기 위해 index에서 Movie App으로 바꿔준다.


3. 영화앱 여러 기능 추가하기-react-router-dom 설치하고 프로젝트 폴더 정리하기

◆네비게이션 기능

Home과 About을 만들어서 각가을 클릭하면 해당 화면으로 이동시키는 메뉴이다.

이렇게 화면을 이동시키기 위해서는 장치가 필요한데 그것을 라우터라고 하며

라우터는 react-router-dom 패키지를 이용하면 쉽게 도입 가능하다.

 

cmd나 터미널에 명령어 입력

npm install react-router-dom

 

 

우선 src폴더 아래에 components폴더랑 routes폴더를 만든다.

그 다음 components에 Movie.css, Movie.js 파일 이동

routes폴더에는 About.js, Home.js를 만들어준다.

그런다음 App.js 코드를 Home.js에 복사하고 이름변경에 맞게 수정해준다.

 

Home.js

import React from 'react';
import axios from 'axios';
import Movie from '../components/Movie';
import './Home.css';

class Home extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };
  getMovies = async () => {
    const {
      data: {
        data: { movies },
      },
    } = await axios.get('https://yts-proxy.now.sh/list_movies.json?sort_by=rating');
    this.setState({ movies, isLoading: false });
  };
  componentDidMount() {
    this.getMovies();
  }
  render() {
    const { isLoading, movies } = this.state;
    return (
      <section className="container">
        {isLoading ? (
          <div className="loader">
            <span className="loader__text">Loading...</span>
          </div>
        ) : (
          <div className="movies">
            {movies.map((movie) => {
              return (
                <Movie
                  key={movie.id}
                  year={movie.year}
                  title={movie.title}
                  summary={movie.summary}
                  poster={movie.medium_cover_image}
                  genres={movie.genres}
                />
              );
            })}
          </div>
        )}
      </section>
    );
  }
}

export default Home;

 

Home.css

.container {
  height: 100%;
  display: flex;
  justify-content: center;
}

.loader {
  width: 100%;
  height: 100vh;
  display: flex;
  justify-content: center;
  align-items: center;
  font-weight: 300;
}
.movies {
  display: grid;
  grid-template-columns: repeat(2, minmax(400px, 1fr));
  grid-gap: 100px;
  padding: 50px;
  width: 80%;
  padding-top: 70px;
}

@media screen and (max-width: 1090px) {
  .movies {
    grid-template-columns: 1fr;
    width: 100%;
  }
}

 

App.js도 수정

import React from 'react';
import Home from './routes/Home';
import './App.css';


function App(){
  return <Home/>;
}

export default App;

4.  영화앱 여러 기능 추가하기-라우터 만들기

라우터는 사용자가 입력한 URL을 통해 특정 컴포넌트를 불러준다.

예를 들어 사용자가 localhost:3000/about이라고 입력하면 About 컴포넌트를 불러주는 것이다.

react-router-dom은 여러 종류의 라우터를 제공하는데 우리는 HashRouter과 Route 컴포넌트를 사용할 것이다.

 

App.js 코드

 

<Route path="/about" element={<About/>}/>

이렇게 작성해주면 localhost:3000/#/about 주소일 경우

About 컴포넌트 화면을 보여준다.

home도 마찬가지

 

 

import React from 'react';
import './App.css';
import {HashRouter,Route, Routes} from 'react-router-dom';
import About from './routes/About';
import Home from './routes/Home';


function App(){
  return(
    <HashRouter>
      <Routes>
      <Route path="/" element={<Home/>}/>
      <Route path="/about" element={<About/>}/>
      </Routes>
    </HashRouter>
  )
}

export default App;

 

About컴포넌트에도 내용이 있어야하므로 간단하게 만들어준다.

 

About.js

import React from 'react';
import './About.css';

function About() {
  return (
    <div className="about__container">
      <span>
        “Freedom is the freedom to say that two plus two make four. If that is granted, all else
        follows.”
      </span>
      <span>- George Orwell, 1984</span>
    </div>
  );
}

export default About;

About.css

.about__container {
  box-shadow: 0 13px 27px -5px rgba(50, 50, 93, 0.25), 0 8px 16px -8px rgba(0, 0, 0, 0.3),
    0 -6px 16px -6px rgba(0, 0, 0, 0.025);
  padding: 20px;
  border-radius: 5px;
  background-color: white;
  margin: 0 auto;
  margin-top: 100px;
  width: 100%;
  max-width: 400px;
  font-weight: 300;
}

.about__container span:first-child {
  font-size: 20px;
}
.about__container span:last-child {
  display: block;
  margin-top: 10px;
}


5. 영화앱 여러 기능 추가하기-내비게이션 만들어보기

라우터가 준비되었으므로 내비게이션을 통해 다른 화면으로 이동하면 된다.

 

먼저 components폴더에 Navigation.js를 만든다.

그리고 Navigation.css 파일도 만든다.

App 컴포넌트에 Navigation 컴포넌트를 import 시킨다

 

페이지 이동시키는 거니까 html a태그 쓰면 되겠지라고 생각하겠지만

a태그는 링크를 누를 때마다 리액트가 죽고 새 페이지가 열리는 문제점이 있어서

react-router-dom의 Link 컴포넌트를 사용한다.

 

Navigation.js

import React from 'react';
import { Link } from 'react-router-dom';
import './Navigation.css';

function Navigation() {
  return (
    <div className="nav">
      <Link to="/">Home</Link>
      <Link to="/about">About</Link>
    </div>
  );
}

export default Navigation;

 

App.js

import React from 'react';
import './App.css';
import {HashRouter,Route, Routes} from 'react-router-dom';
import About from './routes/About';
import Home from './routes/Home';
import Navigation from './components/Navigation';

function App(){
  return(
    <HashRouter>
      <Navigation/>
      <Routes>
      <Route path="/" element={<Home/>}/>
      <Route path="/about" element={<About/>}/>
      </Routes>
    </HashRouter>
  )
}

export default App;

 

Navigation.css

.nav {
    z-index: 1;
    position: fixed;
    top: 50px;
    left: 10px;
    display: flex;
    flex-direction: column;
    background-color: white;
    padding: 10px 20px;
    box-shadow: 0 13px 27px -5px rgba(50, 50, 93, 0.25), 0 8px 16px -8px rgba(0, 0, 0, 0.3),
      0 -6px 16px -6px rgba(0, 0, 0, 0.025);
    border-radius: 5px;
  }
  
  @media screen and (max-width: 1090px) {
    .nav {
      left: initial;
      top: initial;
      bottom: 0px;
      width: 100%;
    }
  }
  
  .nav a {
    text-decoration: none;
    color: #0008fc;
    text-transform: uppercase;
    font-size: 12px;
    text-align: center;
    font-weight: 600;
  }
  
  .nav a:not(:last-child) {
    margin-bottom: 20px;
  }

 


6. 영화앱 여러 기능 추가하기-영화 상세 정보 기능 만들어보기

Home에서 볼 수 있는 영화 정보는 아주 일부분인데 영화 카드를 누르면 상세 정보를 보여주는 기능을 만들 것이다.

 

이 기능을 만들기 위해서는 route props를 알아야한다

route props는 라우팅 대상이 되는 커포넌트에 넘겨주는 기본 props를 말하는데

우리가 직접 넘겨주지 않아도 기본으로 넘어가는 route props라는 것이 있고 이것을 이용해야 영화 데이터를 상세 정보 컴포넌트에 전달할 수 있다.

 

우선 Link를 import 하고

카드를 누르면 상세 정보를 보여줘야하므로 영화 div 태그안에 Link태그로 감싼다.

그리고 Link에 클릭시 state값을 넘겨준다.

import React from "react";
import PropTypes from "prop-types";
import "./Movie.css";
import { Link } from "react-router-dom";  //여기 

function Movie({ title, year, summary, poster, genres }) {
  return (
    <div className="movie">
      <Link //여기
        to={"/movie-detail"}
        state={{ year, title, summary, poster, genres }}
      >
        <img src={poster} alt={title} title={title} />
        <div className="movie__data">
          <h3 className="movie__title">{title}</h3>
          <h5 className="movie__year">{year}</h5>
          <ul className="movie__genres">
            {genres.map((genre, index) => {
              return (
                <li key={index} className="movie__genre">
                  {genre}
                </li>
              );
            })}
          </ul>
          <p className="movie__summary">{summary.slice(0, 180)}</p>
        </div>
      </Link> //여기
    </div>
  );
}

Movie.propTypes = {
  year: PropTypes.number.isRequired,
  title: PropTypes.string.isRequired,
  summary: PropTypes.string.isRequired,
  poster: PropTypes.string.isRequired,
  genres: PropTypes.arrayOf(PropTypes.string).isRequired,
};

export default Movie;

 

카드를 누르면 detail화면에 넘어가게 만들었으니 이제 detail 컴포넌트를 만들자

 

detail 컴포넌트를 우선 간단하게 만들어주고

import React from 'react';
import {useLocation} from "react-router-dom";

function Detail(props){
    const location=useLocation();
    console.log(location);
    return <span>hello</span>;
    }

export default Detail;

 

App.js에 Detail import 추가하고

route path 추가해준다.

import React from 'react';
import './App.css';
import {HashRouter,Route, Routes} from 'react-router-dom';
import About from './routes/About';
import Home from './routes/Home';
import Navigation from './components/Navigation';
import Detail from './routes/Detail'; //여기

function App(){
  return(
    <HashRouter>
      <Navigation/>
      <Routes>
      <Route path="/"  element={<Home/>}/>
      <Route path="/about" element={<About/>}/>
      <Route path="/movie-detail" element={<Detail/>}/> //여기
      </Routes>
    </HashRouter>
  )
}

export default App;

state 값이 잘 들어가져있는 것을 볼 수 있다


7. 영화앱 여러 기능 추가하기-리다이렉트 기능 만들어 보기

영화 카드를 누르면 state가 잘 들어있는데 URL로 직접 /movie-detail 쳐서 들어가면 state값이 null은 것을 확인할 수 있다

Detail 컴포넌트로 영화 데이터가 넘어가지 못했기 때문에 이런 경우 사용자를 강제로 Home으로 돌려보내야한다.

이런 기능을 리다이렉트 기능이라고 부른다.

 

Detail.js 파일을 아래와 같이 바꾼다

import React from 'react';
import {useLocation, useNavigate} from "react-router-dom";
import { useEffect } from 'react';

function Detail(props){
    const location=useLocation();
    const navigate = useNavigate();
    useEffect(() => {
        
        if(location.state===null){
            navigate('/'); //home으로 되돌아 가는 기능
        }
});
   
    if(location.state){
        return <span>{location.state.title}</span> //제목 출력
    }
    return null;
    }

export default Detail;

 

영화앱을 완성했다. 이제 배포해보자.


8. 영화 앱 깃허브에 배포하기

아래 링크보고 따라하면 된다

https://velog.io/@nemo/github-page-deploy-%EB%B0%B0%ED%8F%AC

 


참고문헌

Do it! 클론 코딩 영화 평점 웹서비스-니꼴라스

[React] Link를 통해 State 전달하기 ( Router v6 ) (velog.io)

https://blog.woolta.com/categories/1/posts/211

https://velog.io/@boyfromthewell/React-useLocation-%EC%82%AC%EC%9A%A9%EC%8B%9C-state-%EA%B0%92%EC%9D%B4-null%EB%A1%9C-%EB%93%A4%EC%96%B4%EC%98%A4%EB%8A%94-%EB%AC%B8%EC%A0%9C

 

728x90
LIST
728x90
SMALL

*이 게시글은 Do it! 클론 코딩 영화 평점 웹서비스를 보고 정리한 글입니다.


목차

1. 로딩 구현

2. 영화 API 가져오기

3. 영화 데이터 화면에 그리기

4. 화면 스타일링 하기


1. 로딩 구현

import React from "react";

class App extends React.Component {
  state = {
    isLoading: true,
  };

  render() {
    const { isLoading } = this.state;  //구조분해할당
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>; //삼항연산자
  }
}

export default App;

구조분해할당을 모르면 이해 자체가 안될 것이다

구조 분해 할당 - JavaScript | MDN (mozilla.org)

 

위 설명을 잘 참고하길 바란다.

 

state 내부의 isLoading을 개별 변수로 사용하기 위해 구조분해할당 기법을 사용하였다. 

구조분해할당한 덕분에 isLoading을 변수로 쓸 수 있게 되었다.

 

삼항연산자는

isLoading이 True면 Loading이 출력되고 False면 We are ready가 출력된다.

 

영화를 담아야하니 state에 movies라는 배열도 추가한다

import React from "react";

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  render() {
    const { isLoading } = this.state;
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>;
  }
}

export default App;

2. 영화 API 가져오기

우선 render()함수가 실행되면 호출되는 생명주기 함수가 바로 componentDidMount()함수이다.

이 함수를 이용해서 렌더링된 후 바로 화면에 나와야할 영화데이터를 넣을 것이다.

 

원래는 자바스크립트의 fetch() 함수를 알아야하지만 초보자가 사용하기는 난이도가 있어서 Axios라는 도구를 사용한다.

npm install axios

API Documentation - YTS YIFY

영화 데이터를 위 사이트에서 가져올 것이다

 

아래와 같이 App.js 파일을 수정한다.

화면에는 Loading...만 나오지만 오류가 나오지 않으므로 axios는 잘 작동하는 것이다.

import React from "react";
import axios from 'axios';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };
  componentDidMount(){
    axios.get('https://yts-proxy.now.sh/list_movies.json');
  }

  render() {
    const { isLoading } = this.state;
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>;
  }
}

export default App;

브라우저에 F12를 눌러서 나오는 관리자모드에 네트워크에 들어가면

list_movies.json이라는 것이 보이는데 이게 axios가 API를 호출하고 있기 때문에 생긴 것이다

 

axios는 네트워크를 사용하기 때문에 느리게 작동한다.

그래서 axios.get()이 반환한 영화 데이터를 잡으려면 자바스크립트에서 axios.get()을 포함하고 있는 함수의 실행이 끝날 때 까지 시간이 걸릴 수 있다고 알려줘야한다.

 

getMovies라는 함수를 만들고 그 함수 안에서 axios.get()이 실행되도록 만든다.

그리고 axios.get()이 반환한 결과를 movies에 저장한다.

import React from "react";
import axios from 'axios';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=()=>{
    const movies =axios.get('https://yts-proxy.now.sh/list_movies.json');
  }

  componentDidMount(){
    this.getMovies();
  }

  render() {
    const { isLoading } = this.state;
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>;
  }
}

export default App;

 

이렇게 하면 componentDidMount()함수가 실행되면 this.getMovies()가 실행될 것이다.

이때 자바스크립트에게 getMovies()함수는 시간이 좀 필요하다고 말해야만

axios.get()이 반환한 데이터를 제대로 잡을 수 있다.

그것을 하기 위해서 async, await라는 키워드가 필요하다.

 

자바스크립트에게 getMovies()함수는 시간이 필요하다고 알리기위해 async를 쓰고

axios.get()의 실행을 기다려 달라고 말해주기 위해 await가 필요하다

import React from "react";
import axios from 'axios';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=async ()=>{
    const movies =await axios.get('https://yts-proxy.now.sh/list_movies.json');
  }

  componentDidMount(){
    this.getMovies();
  }

  render() {
    const { isLoading } = this.state;
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>;
  }
}

export default App;

 

중간정리

리액트 앱이 실행되면 최초로 render()함수가 실행되고 최초의 state에는 isLoading, movies가 있다.

isLoading은 true이고 movies는 빈배열이다

 

이어서 App 컴포넌트가 마운트되면 componentDidMount() 함수가 실행되면서 getMovies() 함수가 실행된다.

여기서 getMovies()함수에 시간이 많이 걸리는 axios.get()이 포함돼있어 async와 awawit를 붙여준다.


3. 영화 데이터 화면에 그리기

 

console.log(movies)라고 해보자.

import React from "react";
import axios from 'axios';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=async ()=>{
    const movies =await axios.get('https://yts-proxy.now.sh/list_movies.json');
    console.log(movies);
    
  }

  componentDidMount(){
    this.getMovies();
    
  }

  render() {
    const { isLoading } = this.state;
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>;
  }
}

export default App;

콘솔에 보면 우리가 영화 사이트를 만들기 위해 필요한 정보들이 들어있다.

data->data->movies 순서대로 개갸체에 접근하면 원하는 데이터를 추출할 수 있을 것이다

 

console.log(movies.data.data.movies)라고 해보자.

import React from "react";
import axios from 'axios';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=async ()=>{
    const movies =await axios.get('https://yts-proxy.now.sh/list_movies.json');
    console.log(movies.data.data.movies);
    
  }

  componentDidMount(){
    this.getMovies();
    
  }

  render() {
    const { isLoading } = this.state;
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>;
  }
}

export default App;

 

원하는대로 원하는 영화 데이터만 추출할 수 있게 된 것이다

movies.data.data.movies로 객체에 접근하는 것은 아름답지않아서 아래와같이 구조분해할당으로 코드를 수정한다.

import React from "react";
import axios from 'axios';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=async ()=>{
    const {data:{
      data:{
        movies
      }
    }} =await axios.get('https://yts-proxy.now.sh/list_movies.json');
    console.log(movies);
    
  }

  componentDidMount(){
    this.getMovies();
    
  }

  render() {
    const { isLoading } = this.state;
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>;
  }
}

export default App;

똑같은 결과인 것을 볼 수 있다.

 

데이터 추출을 했으니 이제 이 영화 데이터를 state에 저장하면 될 것이다.

console.log를 지우고 this.setState({movies:movies});를 입력한다.

 

setState는 State 값을 변경해준다.

빈 배열이던 movies state에 movies(영화 데이터)를 넣어준다.

import React from "react";
import axios from 'axios';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=async ()=>{
    const {data:{
      data:{
        movies
      }
    }} =await axios.get('https://yts-proxy.now.sh/list_movies.json');
    this.setState({movies:movies});
  }

  componentDidMount(){
    this.getMovies();
  }

  render() {
    const { isLoading } = this.state;
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>;
  }
}

export default App;

그런데 자바스크립트 ES6에서는 객체의 키와 대입할 변수의 이름이 같다면 코드를 축약할 수 있다.

this.setState({movies:movies});를

this.setState({movies})로 쓸 수 있는 것이다

import React from "react";
import axios from 'axios';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=async ()=>{
    const {data:{
      data:{
        movies
      }
    }} =await axios.get('https://yts-proxy.now.sh/list_movies.json');
    this.setState({movies});
  }

  componentDidMount(){
    this.getMovies();
    
  }

  render() {
    const { isLoading } = this.state;
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>;
  }
}

export default App;

 

영화데이터를 받았으니 Loading화면이 아니라 We are ready가 출력돼야할 것이다.

따라서 setState에 isLoading을 false로 바꾸는 내용을 추가한다.

 this.setState({movies, isLoading: false});

import React from "react";
import axios from 'axios';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=async ()=>{
    const {data:{
      data:{
        movies
      }
    }} =await axios.get('https://yts-proxy.now.sh/list_movies.json');
    this.setState({movies, isLoading: false});
    
  }



  componentDidMount(){
    this.getMovies();
    
  }

  render() {
    const { isLoading } = this.state;
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>;
  }
}

export default App;

 

영화데이터를 받았고 영화데이터를 받은 후 isLoading, Movies state값을 변경해줬지만 화면에는 영화 데이터가 아닌 We are ready만 뜬다.

우리는 We are ready가 아니라 영화 정보가 렌더링돼야 하므로 movies state를 그리기 위해 Movies 컴포넌트를 만든다.

 

Movie.js라는 파일을 만든다.

Movie 컴포넌트는 state가 필요하지 않으므로 함수형 컴포넌트로 만든다.

 

아래와 같이 뼈대를 만든다.

import React from 'react';
import PropTypes from 'prop-types';

function Movie(){
 return <h1></h1>;
}


Movie.propTypes = {};

export default Movie;

 

위 링크에서 필요한 데이터가 뭔지 일단 확인한다.

아래와 같이 안뜨면 크롬브라우저 스토어에서 JSON 뷰어를 설치하기 바란다.

필요한 데이터를 확인하고

개발자가 실수하는 것을 방지하기 위해 우선 propTypes를 작성한다.

주의할 것은 poster에는 image의 주소가 들어간다

API에서는 medium_cover_image라고 했지만 이해하기 쉽도록 poster라고 지정했다.

import React from 'react';
import PropTypes from 'prop-types';


function Movie() {
  return (
    <div>
    
    </div>
  );
}

Movie.propTypes = {
  id : PropTypes.number.isRequired,
  year: PropTypes.number.isRequired,
  title: PropTypes.string.isRequired,
  summary: PropTypes.string.isRequired,
  poster: PropTypes.string.isRequired,
};

export default Movie;

그냥 영화를 보여주면 재미없으므로 랭킹 순서대로 보여주기 위해 axios.get()에 주소를 수정한다.

await axios.get('https://yts-proxy.now.sh/list_movies.json?sort_by=rating');
import React from "react";
import axios from 'axios';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=async ()=>{
    const {data:{
      data:{
        movies
      }
    }} =await axios.get('https://yts-proxy.now.sh/list_movies.json?sort_by=rating');
    this.setState({movies, isLoading: false});
  }

  componentDidMount(){
    this.getMovies();
  }

  render() {
    const { isLoading } = this.state;
    return <div>{isLoading ? "Loading..." : "We are ready"}</div>;
  }
}

export default App;

 

이제 평점 내림차순으로 영화 데이터를 가져올 수 있게 되었으므로 App 컴포넌트에서 Movie 컴포넌트로 id, title, year, summary ,poster props를 넘겨주면 된다. 

 

아래와 같이 구조 분해 할당으로 props를 받는다.

import React from 'react';
import PropTypes from 'prop-types';


function Movie({ id,title, year, summary, poster }) {
  return (
    <div >
      <img src={poster} alt={title} title={title} />
      <div >
        <h3 >{title}</h3>
        <h5 >{year}</h5>
        <p >{summary}</p>
      </div>
    </div>
  );
}

Movie.propTypes = {
  id : PropTypes.number.isRequired,
  year: PropTypes.number.isRequired,
  title: PropTypes.string.isRequired,
  summary: PropTypes.string.isRequired,
  poster: PropTypes.string.isRequired,
};

export default Movie;

 

그 후 App.js(App 컴포넌트)로 가서 영화 데이터에 props를 전달할 수 있도록 수정한다.

 

제일 먼저 Movie component import(가져오기)를 한다.

import Movie from'./Movie';

 

그다음 We are ready를 지우고 movies.map()으로 변경한다.

movie라는 변수에 movies 배열의 요소 하나하나를 다 저장하고 콘솔화면에 그 변수를 출력하는 코드로 수정한다.

그 후 Movie 컴포넌트를 리턴한다.

Movie 컴포넌트를 리턴하긴 했지만 props를 전달하지 않아 아무것도 출력되지 않는다.

return <div>{isLoading ? "Loading..." : movies.map(
      (movie)=>{
        console.log(movie);
        return <Movie />;
    }
import React from "react";
import axios from 'axios';
import Movie from'./Movie';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=async ()=>{
    const {data:{
      data:{
        movies
      }
    }} =await axios.get('https://yts-proxy.now.sh/list_movies.json?sort_by=rating');
    this.setState({movies, isLoading: false});
  }

  componentDidMount(){
    this.getMovies();
  }

  render() {
    const { isLoading, movies } = this.state;
    return <div>{isLoading ? "Loading..." : movies.map(
      (movie)=>{
        console.log(movie);
        return <Movie />;
    }
    )}</div>;
  }
}

export default App;

 

Movie 컴포넌트에 props를 전달한다.

   return <Movie 
        id={movie.id}
        year={movie.year}
        title={movie.title}
        summary={movie.summary}
        poster={movie.medium_cover_image}

        
        />;
import React from "react";
import axios from 'axios';
import Movie from'./Movie';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=async ()=>{
    const {data:{
      data:{
        movies
      }
    }} =await axios.get('https://yts-proxy.now.sh/list_movies.json?sort_by=rating');
    this.setState({movies, isLoading: false});
  }

  componentDidMount(){
    this.getMovies();
  }

  render() {
    const { isLoading, movies } = this.state;
    return <div>{isLoading ? "Loading..." : movies.map(
      (movie)=>{
        console.log(movie);
        return <Movie 
        id={movie.id}
        year={movie.year}
        title={movie.title}
        summary={movie.summary}
        poster={movie.medium_cover_image}

        
        />;
    }
    )}</div>;
  }
}

export default App;

 아래와 같이 화면에 잘 뜨는 것을 볼 수 있다.

 

마지막으로

 

App 컴포넌트에 

console.log(movie)는 필요없으니 지우고 key 값을 추가한다.

 

아래와 같이 컴포넌트를 여러개 출력할 때는 유일한 값을 이용하여 key props를 추가해야한다.

   return <div>{isLoading ? "Loading..." : movies.map(
      (movie)=>{
        return <Movie 
        key={movie.id} //이 부분
        id={movie.id}
        year={movie.year}
        title={movie.title}
        summary={movie.summary}
        poster={movie.medium_cover_image}
        />;
    }
    )}</div>;

 

import React from "react";
import axios from 'axios';
import Movie from'./Movie';

class App extends React.Component {
  state = {
    isLoading: true,
    movies: [],
  };

  getMovies=async ()=>{
    const {data:{
      data:{
        movies
      }
    }} =await axios.get('https://yts-proxy.now.sh/list_movies.json?sort_by=rating');
    this.setState({movies, isLoading: false});
  }

  componentDidMount(){
    this.getMovies();
  }

  render() {
    const { isLoading, movies } = this.state;
    return <div>{isLoading ? "Loading..." : movies.map(
      (movie)=>{
        return <Movie 
        key={movie.id}
        id={movie.id}
        year={movie.year}
        title={movie.title}
        summary={movie.summary}
        poster={movie.medium_cover_image}
        />;
    }
    )}</div>;
  }
}

export default App;

출처

Do it! 클론 코딩 영화 평점 웹서비스-니꼴라스

 

728x90
LIST
728x90
SMALL

*아래 연습문제는 연습문제를 위한 견본데이터베이스가 있어야 실행 가능

★연습문제1

1. 회원관리(EC_Member) 테이블을 이용하여 총회원수를 출력하시오.

더보기
SELECT COUNT(*)"총회원수"
FROM   EC_Member;

2. 주문처리(EC_Order) 테이블에서 결제금액(CMONEY)이 제일 적은 금액과 제일 많은 금액을 출력하시오.

더보기
SELECT MIN(CMONEY),MAX(CMONEY)
FROM   EC_Order;

3. 주문처리(EC_Order) 테이블에서 결제한 회원의 주문자ID별 결제 금액(CMONEY)의 합계를 계산하여 [주문자ID, 결제합계]를 출력하시오.

더보기
SELECT   ORDER_ID "주문자 ID", TO_CHAR(SUM(CMONEY),'L999,999,999')"결제합계"
FROM     EC_Order
GROUP BY ORDER_ID;

4. 상품관리(EC_Product) 테이블의 단가(Unit_Price)를 이용하여 역순으로 상품 단가에 순위를 부여하여 [상품코드, 상품명, 단가, 순위]를 출력하시오.

더보기
SELECT PRODUCT_CODE, PRODUCT_NAME, UNIT_PRICE,
         RANK() OVER (ORDER BY UNIT_PRICE DESC) "상품단가순위"
FROM     EC_Product
ORDER BY 3 DESC;

5. 회원관리(EC_Member) 테이블을 이용하여 회원의 시도별 분포 인원수와 전체 인원수를 계산하여 출력하시오. (ROLLUP 연산)

더보기
SELECT UNIQUE SUBSTR(ADDRESS,1,2)"시도별",
COUNT(*) "인원수"
FROM EC_Member
GROUP BY ROLLUP(SUBSTR(ADDRESS,1,2))
ORDER BY 1;

6. 주문처리(EC_Order) 테이블에서 결제 년월별 판매금액계를 출력하시오. GROUPING() 함수를 사용하여 '결제년월'이 연산에 사용되지 않을 경우 '합계금액'을 출력한다. (ROLLUP 연산, CASE() 함수 사용)

더보기
SELECT CASE GROUPING(SUBSTR(CDATE,1,5)) WHEN 0 THEN SUBSTR(CDATE,1,5)
                                        WHEN 1 THEN ' 합계금액'
       END "결제년월",
       TO_CHAR(SUM(CMONEY),'L999,999,999') "판매금액계"
FROM EC_Order
GROUP BY ROLLUP(SUBSTR(CDATE,1,5))
ORDER BY GROUPING(SUBSTR(CDATE,1,5)),1;

7. 주문처리(EC_Order) 테이블에서 2018년 1/4분기(1월~3월)의 주문자ID, 주문상품, 결제금액을 (주문자ID, 상품코드별), 회원별 집계, 상품코드별 집계, 전체 합계를 계산하여 주문자ID 순으로 출력하시오. (CUBE 연산, CASE() 함수, GROUPING_ID() 함수 사용)

더보기
SELECT CASE GROUPING_ID(ORDER_ID, PRODUCT_CODE) WHEN 0 THEN ORDER_ID
                                                WHEN 1 THEN NULL
                                                WHEN 2 THEN ' 상품 집계'
                                                WHEN 3 THEN ' 전체 합계'
       END "주문자_ID",
       CASE GROUPING_ID(ORDER_ID, PRODUCT_CODE) WHEN 0 THEN PRODUCT_CODE
                                                WHEN 1 THEN ' 회원집계'
                                                WHEN 2 THEN PRODUCT_CODE
                                                WHEN 3 THEN NULL
                                                
                                       
       END "주문상품", TO_CHAR(SUM(CMONEY),'L999,999,999') "결제금액"
FROM   EC_Order
WHERE TO_CHAR(CDATE,'YY/MM') >='18/01' AND TO_CHAR(CDATE,'YY/MM')<='18/03'
GROUP  BY CUBE(ORDER_ID,PRODUCT_CODE)
ORDER BY ORDER_ID;

★연습문제2

1. 주문처리(EC_Order) 테이블에서 결제한 결제년월별로 [결제년월, 주문건수, 주문 수량 합계, 결제 금액 합계]를 계산하여 결제년월순으로 출력하시오.

더보기
SELECT TO_CHAR(Cdate, 'YYYY/MM') "결제년월", Count(2) "주문건수",
       SUM(Order_QTY) "주문수량 합계", TO_CHAR(SUM(CMONEY),'L999,999,999')"결제합계"
FROM EC_ORDER
WHERE Cdate IS NOT NULL
GROUP BY TO_CHAR(Cdate, 'YYYY/MM')
ORDER BY 1;

2. 회원관리(EC_Member) 테이블에서 남, 여 회원수를 출력하시오. 단, 성별 코드가 1또는 3이면 남자, 2 또는 4는 여자임.

더보기
SELECT DECODE(SUBSTR(REGIST_NO,8,1),1,'남자',2,'여자',3,'남자',4,'여자')
"성별", COUNT(*) "회원수"
FROM EC_MEMBEr
GROUP BY DECODE(SUBSTR(REGIST_NO,8,1),1,'남자',2,'여자',3,'남자',4,'여자');

3. 주문처리(EC_Order) 테이블에서 년월별 결제금액이 가장 많은 금액을 결제년월별순으로 출력하시오.

더보기
SELECT TO_CHAR(Cdate, 'YYYY/MM') "결제년월", TO_CHAR(MAX(CMoney),'L999,999,999') "결제최대금액"
FROM   EC_ORDER
WHERE  CDATE IS NOT NULL
GROUP  BY TO_CHAR(Cdate, 'YYYY/MM')
ORDER BY 1;

4. 주문처리(EC_Order) 테이블에서 주문자ID, 주문상품코드, 구매횟수, 결제금액을 (주문자, 상품코드별), 주문자별 소계, 전체 합계를 계산하여 주문자ID순으로 출력하시오. (ROLLUP 연산, CASE() 함수와 GROUPING_ID() 함수 사용)

더보기
SELECT CASE GROUPING_ID(Order_ID, Product_Code) WHEN 1 THEN NULL
                                                       ELSE Order_ID
            END "주문자_ID",
            CASE GROUPING_ID(Order_ID, Product_Code) WHEN 1 THEN '     소계'
                                                    WHEN 3 THEN '전체합계'
                                                        ELSE Product_Code
            END "주문상품",
            COUNT(*) "구매횟수", TO_CHAR(SUM(Cmoney),'L999,999,999')"결제금액"
FROM       EC_Order
WHERE      CDate IS NOT NULL
GROUP      BY ROLLUP(Order_ID, Product_Code)
ORDER      BY ORder_ID;

5. 주문처리(EC_Order) 테이블에서 주문자ID, 상품코드, 결제금액을 주문자별, 상품코드별로 계산하여 주문자, 상품코드순으로 출력하시오. (GROUPING SETS 연산)

더보기
SELECT Order_ID, Product_Code, TO_CHAR(SUM(Cmoney),'L999,999,999') "결제금액"
FROM   EC_Order
WHERE CDAte IS NOT NULL
GROUP BY GROUPING SETS(Order_ID, Product_Code)
ORDER BY 1,2;

6. 주문처리(EC_Order) 테이블에서 결제자의 결제금액을 이용하여 높은 금액부터 순위를 구하여 [주문자ID, 결제일자, 결제방법, 결제금액, 순위]를 출력하시오.

더보기
SELECT Order_ID, CDATE "결제일자", Csel "결제방법", TO_CHAR(CMoney,'L999,999,999') "결제금액",
       DENSE_RANK() OVER(ORDER BY CMoney DESC) "순위"
FROM   EC_Order
WHERE  CDate IS NOT NULL
ORDER BY 4 DESC;

7. 수강임시(T_SG_Scores) 테이블에서 성적 취득한 학번별 과목수, 합계와 평균을 계산하여 [학번, 과목수, 총점, 평균]을 총점 내림차순으로 출력하시오.

더보기
SELECT Student_ID, COUNT(1) "과목수", SUM(Score) "총점", AVG(Score) "평균"
FROM   T_SG_Scores
WHERE  Score IS NOT NULL
GROUP  BY STudent_ID
ORder  BY 3 DESC;

8. 수강(SG_Scores) 테이블의 학번별 총점과 평균을 구하고, 평균으로 석차를 구하여 [학번, 총점, 평균, 석차]를 석차순으로 출력하시오.

더보기
SELECT Student_ID, SUM(Score) "총점", TO_CHAR(AVG(Score), '99.9') "평균",
       RANK()  OVER(ORDER BY AVG(Score) DESC) "석차"
FROM   SG_Scores
GROUP  BY Student_ID
ORDER  BY 4;

9. 자유게시판(Fress_Board) 테이블의 게시물 행을 수를 출력하고, 게시물의 저장하기 위해 게시판 번호(B_ID)를 구하시오. 단, 게시물 번호는 널이면 1, 그렇지 않으면 게시물번호의 최댓값+1로 한다. (CASE문 사용)

더보기
SELECT COUNT(*) "총게시물수" FROM FRee_Board;
SELECT CASE WHEN MAX(B_ID) IS NULL THEN 1
                                    ELSE MAX(B_ID)+1
       END "게시물번호"
FROM   Free_BOARD;

 

728x90
LIST
728x90
SMALL

*이 게시글은 Do it! 클론 코딩 영화 평점 웹서비스를 보고 정리한 글입니다.


목차

1. create-react-app으로 리액트 폴더 생성

2. 파일 수정

3. git허브에 리액트 앱 업로드 하기

4. src 폴더 정리하기, index.js, App.js 수정


Movie App (nomadcoders.github.io)

우리가 만들고자 하는 사이트는 위와 같은 영화 평점 웹서비스 사이트이다.


1.  create-react-app으로 리액트 폴더 생성

원하는 위치에 이동후 아래와 같은 명령어를 cmd나 shell, 터미널에 입력

보통 폴더를 하나 만들고 그 위치에 만든다.

npx create-react-app movie_app_2020

다 완료되면 아래와 같은 메시지와 함께 폴더에 여러 파일들이 들어가진다.

아래와 같은 명령어를 또 친다.

 cd movie_app_2020
 npm start

그러면 아래와 같은 화면이 뜰 것이다.

준비단계 끝

터미널 창을 끄면 리액트 앱 연결도 끊기니 터미널은 계속 켜놔야한다.

 

다음으로 해당 폴더를 비주얼코드로 들어간다.

아래와 같이 많은 파일이 보인다.


2. 파일 수정

README.md 파일 다 지우고 아래와 같이 입력

package.json파일에 test, eject 삭제


3. git허브에 리액트 앱 업로드 하기

git없으면 git 설치 먼저 하기

Git - Downloads (git-scm.com)

 

1) 명령어 입력

cd movie_app_2020 
git init

 

2) 깃허브에 저장소 만들기

New repository · GitHub

회원 가입 후 위 사이트 접속

 

아래와 같이 입력후 create repository입력

.git으로 끝나는 주소가 저장소 주소이다.

.ㅎ

 

3) 아래와 같이 명령어 입력

git add .
git commit -m "02-2깃허브에 리액트 앱 업로드하기"
git branch -M main
git remote add origin https://github.com/windbrain/movie_app_2020.git
git push -u origin main

 

4) 잘 생성된 것을 볼 수 있다.


4. src 폴더 정리하기, App.js 수정

App.css

App.test.js

index.css

logo.svg

reportWebBitals.js

setupTests.js

package-lock.json

삭제

index.js 정리

import React from 'react';
import ReactDOM from 'react-dom/client';
import App from './App';


const root = ReactDOM.createRoot(document.getElementById('root'));
root.render(
    <App />
);

 

App.js 정리

function App() {
  return <div className="App"/>;
}
export default App;

 

정리하고 화면 확인

아무것도 안나오는 것이 정상


출처

Do it! 클론 코딩 영화 평점 웹서비스-니꼴라스

728x90
LIST
728x90
SMALL

*이 게시글은 인프런-처음 만난 리액트 강좌를 보고 정리한 글입니다.


목차

1. Component

2. Props

3. Props의 특징

4. Props의 사용법

5. Component 만들기

6. Component 렌더링

7. Component 합성

8. Component 추출


1. Component

리액트는 Component 기반구조라는 것을 배웠었다.

리액트는 모든 페이지가 컴포넌트로 구성돼있고 그 컴포넌트는 또다른 여러개 컴포넌트의 조합으로 이뤄져있다.

이런 식으로 레고 블록을 조립하듯 컴포넌트들을 모아서 개발할 수 있는 것이다

 

아래가 바로 컴포넌트로 페이지를 만든 하나의 예시이다.

출처 : 처음 만난 리액트-인프런 강좌

 

리액트의 컴포넌트는 개념적으로는 자바스크립트의 함수와 비슷하다.

출처 : 처음 만난 리액트-인프런 강좌

기본 메커니즘은 자바스크립트의 함수와 동일하지만 입, 출력이 조금 다르다.

 

리액트 컴포넌트에서 입력은 바로 다음에 배울 Props라는 것이고

출력은 앞에서 배운 React Element가 된다.

 

즉 리액트는 어떠한 속성을 입력받아서 그에 맞는 element를 생성하여 리턴해주는 것이다.

출처 : 처음 만난 리액트-인프런 강좌

이는 마치 붕어빵틀을 통해 붕어빵을 만드는데 슈크림, 팥, 고구마등의 앙코(팥소)를 넣어서 각각의 붕어빵을 만드는 것과 비슷하다.

슈크림, 팥, 고구마 등의 앙코가 props(속성)

붕어빵틀이 Component

만들어진 붕어빵이 element인 것이다.

출처 : 처음 만난 리액트-인프런 강좌


2. Props

props는 prop이 여러개인 것을 의미하는데 prop는 Property를 줄여서 쓴 것으로 

속성이라는 의미이다.

바로 컴포넌트의 속성이다.

아래 붕어빵 그림을 보면 쉽게 이해할 수 있다.

앙코가 props인 것이다.

출처 : 처음 만난 리액트-인프런 강좌

 

실제 예시로 Props를 알아보자.

출처 : 처음 만난 리액트-인프런 강좌

위 컴포넌트를 A컴포넌트라고 하면 다 같은 컴포넌트에서 생성된 녀석들이지만 각기 다른 이미지와 텍스트를 가지고 있는 것은 아래와 같이 Props가 다르기 때문이다.

출처 : 처음 만난 리액트-인프런 강좌

 

Props를 다시 정리하면 다음과 같다.

컴포넌트에 전달할 다양한 정보를 담고 있는 자바스크립트 객체인 것이다. 

 

우리가 컴포넌트에 데이터를 전달하고 전달된 데이터에 따라 다른 모습의 엘리먼트를 렌더링하고 싶을 때 해당 데이터를 부모 컴포넌트에서  props에 넣어서 전달하는 것이다.

출처 : 처음 만난 리액트-인프런 강좌


3. Props의 특징

1) 읽기전용(Read-Only)

읽기 전용이라는 것은 값을 변경할 수 없다는 것이다.

출처 : 처음 만난 리액트-인프런 강좌

따라서 새로운 Props를 Component에 전달해줘야한다. 

출처 : 처음 만난 리액트-인프런 강좌

 

제대로 이해하기 위해서는 자바스크립트 함수의 속성에 대해 알아야한다

출처 : 처음 만난 리액트-인프런 강좌

위의 함수는 입력값이 바로 return이 되므로 입력값이 바뀌지 않는다. 이런 것을 Pure(순수)하다고 한다.

 

아래의 함수는 입력값이 바뀌게 되므로 Impure(순수x)하다고 한다.

출처 : 처음 만난 리액트-인프런 강좌

React의 공식 문서에는 다음과 같이 설명하는데

모든 리액트 컴포넌트는 Props를 직접 바꿀 수 없고

같은 Props에 대해서는 항상 같은 결과를 보여줘야한다는 의미이다.

 

출처 : 처음 만난 리액트-인프런 강좌


4. Props의 특징

아래의 코드 같은 경우는 JSX코드인데

아래 그림과 같이 

키=값 형태로 props를 넣을 수 있다.

 

App 컴포넌트가 나오고

그 안에 Profile 컴포넌트가 나오는데

그 컴포넌트의 Props의 값을 넣어주는 형태이다.

 

Props의 문자열 외의 값을 넣을 때(정수, 변수, 다른 컴포넌트)는 중괄호를 통해 감싸줘야한다.

출처 : 처음 만난 리액트-인프런 강좌

props를 전달하면 아래와 같은 자바스크립트 객체가 된다.

출처 : 처음 만난 리액트-인프런 강좌

 

그리고 다음과 같이 Props에 중괄호를 사용해 다른 컴포넌트도 넣을 수 있다. 

출처 : 처음 만난 리액트-인프런 강좌

 

JSX코드를 사용하지 않은 코드는 다음과 같다.

출처 : 처음 만난 리액트-인프런 강좌
출처 : 처음 만난 리액트-인프런 강좌



5. Component 만들기

리액트 컴포넌트는 아래 그림과 같이 Function Component(함수 컴포넌트)와 Class Component(클래스 컴포넌트)로 나뉜다. 

리액트의 초기 버전에서는 Class Component를 많이 사용하였으나 쓰기 불편하다는 이야기가 많아 Function Component를 더 많이 사용하게 되었다.

함수 컴포넌트를 개선하는 과정에서 만들어진 것이 Hook이라는 개념이다. 

출처 : 처음 만난 리액트-인프런 강좌

 

1) 함수 컴포넌트

아래와 같은 형태가 함수 컴포넌트로 Welcome 컴포넌트 안에 저렇게 props에 name을 '바보'라고 키=값 형태로 props를 넣으면 {props.name}자리에 바보라고 출력되는 것을 볼 수 있다.

const root = ReactDOM.createRoot(document.getElementById('root'));
root.render(

    <Welcome name="바보"/>
 
);

출처 : 처음 만난 리액트-인프런 강좌

화살표함수로 나타내면 아래와 같이 쓸 수 있다.

const Welcome=(props)=>{
    return <h1>안녕, {props.name}</h1>;
}

2) 클래스 컴포넌트

클래스 컴포넌트는 함수 컴포넌트에 비해서 몇가지 추가적인 기능을 가진다.

 

아래의 클래스 컴포넌트 코드는 위의 함수 컴포넌트와 동일한 기능을 수행한다.

모든 클래스 컴포넌트는 Component를 상속받아서 만들어진다.

 

아래와 같이 Component를 import시켜야 사용가능하다.

import {Component} from 'react';

 

*상속은 객체지향에서 나오는 개념으로 한 클래스에 변수들과 함수들을 상속받아서 새로 자식 클래스를 만드는 방법이다.

 

★컴포넌트를 만들 때 주의점

Component 이름은 항상 대문자로 시작해야한다

React는 소문자로 시작하는 component를 DOM 태그로 인식하기 때문이다.

출처 : 처음 만난 리액트-인프런 강좌


6. Component 렌더링

아래 코드는

Welcome 함수 컴포넌트를 선언하고

element를 생성한 후 생성된 element를 root div태그에 렌더링을 한다.

따라서 name="인제"라는 props값이 props.name이 전달된다. 

출처 : 처음 만난 리액트-인프런 강좌


7. Component 합성

Component안에는 또 다른 Component를 쓸 수 있기 때문에 복잡한 화면을 여러 개의 Component로 나눠서 구현 가능하다.

 

아래는 Welcome 컴포넌트의 props를 다르게 해서 Welcome 컴포넌트를 여러번 사용하는 것을 볼 수 있다.

이렇게 여러개의 Component를 합쳐서 또 다른 Component를 만드는 것을 Component 합성이라고 한다.

출처 : 처음 만난 리액트-인프런 강좌

 

아래와 같은 형태일때 App Component가 Welcome Component의 부모 컴포넌트인데

부모 컴포넌트에서 아래와 같이 props값을 설정할 수 있다.

출처 : 처음 만난 리액트-인프런 강좌


8. Component 추출

component합성과는 반대로 복잡한 Component를 여러개의 Component로 쪼개는 것을 Component 추출이라고 한다.

Component 추출을 잘 활용하게 되면 재사용성이 올라가는데  컴포넌트가 작아질 수록 컴포넌트의 기능과 목적이 명확해지고 props도 단순해지기 때문에 다른 곳에서 사용할 수 있을 확률이 높아지기 때문이다.

재사용성이 올라가기 때문에 개발속도도 빨라진다.

 

아래 코드의 정확한 분석이 아닌 흐름만 봐보도록 하자.

아래와 같은 코드가 있다.

Component 추출을 해보자.

 

아래와 같이 Avator 컴포넌트로 추출을 했다.

아래와 같이 전보다 깔끔해진 것을 볼 수 있다.

 

마찬가지로 아래코드를 추출했다.

 

아래와 같이 코드가 직관적이고 가독성도 높게 바뀌었다.

최종적으로 아래와 같은 형태가 되었다.


참고문헌

1. [무료] 처음 만난 리액트(React) - 인프런 | 강의 (inflearn.com)

728x90
LIST
728x90
SMALL

*이 게시글은 인프런-처음 만난 리액트 강좌를 보고 정리한 글입니다.


목차

1. Elements란?

2. createElement가 동작하는 과정

3. React elements의 특징

4. Elements 렌더링하기


1. Elements란?

Elements의 영단어는 어떤 물체를 구성하는 요소, 성분이라는 뜻인데

마찬가지로 여기서 배우는 Elements도 리액트 앱을 구성하는 요소를 뜻한다.

 

아래는 공식사이트에서 elements를 설명하고 있는 문장인데 Elements는 리액트 앱에서 가장 작은 블록들이라고 적혀있다.

출처 : 처음 만난 리액트-인프런 강좌

 

이렇게 Chrome 브라우저에서 개발자도구의 Elements에서 볼 수 있는 것은 React Elements가 아니라 Dom Elements이다.

출처 : 처음 만난 리액트-인프런 강좌

 

리액트를 개발하기 시작한 아주 초장기에 화면에 나타나는 내용을 기술하는 자바스크립트 객체를 일컫는 용어가 필요했는데 처음에는 Descriptor이라고 불리다가 최종적으로 나타나는 형태는 DOM Elements여서

통일성을 위해 똑같이 Elements라 부르기로 결정했다.

출처 : 처음 만난 리액트-인프런 강좌

즉, 아래와 같이 createElement(현재는 다른 방식으로 사용됨)의 결과로 객체가 생성되는데 그게 element이다.

출처 : 처음 만난 리액트-인프런 강좌

React elements는 앞에서 배웠던 Virtual DOM의 Elements를 나타낸다.

출처 : 처음 만난 리액트-인프런 강좌

Elements는 화면에서 보이는 것들을 기술하는데 아래와 같은 JSX코드를 작성하면 내부적으로는 JS코드로 변환하여

React Elements가 생성된다 

Element는 HTML에서 태그로 적은 노드들을 지칭하므로 가상DOM에 <h1> element가 만들어지는 것이다.

출처 : 처음 만난 리액트-인프런 강좌

 

이 React Elements는 앞에서도 말했지만 자바스크립트 객체 형태로 존재하며 

마음대로 변경할 수 없는 불변성을 가지고 있다

 

출처 : 처음 만난 리액트-인프런 강좌

아래는 button을 나타내기위한 Element이다

(JSX코드가 JS코드로 바뀌면서 생성되는 자바스크립트 객체인 element)

출처 : 처음 만난 리액트-인프런 강좌

위 코드를 렌더링하면 아래와 같은 DOM element가 된다

출처 : 처음 만난 리액트-인프런 강좌


2. createElement의 동작과정

다시 복습을 하면 아래와 같은 JSX코드를 내부적으로는 바벨이 JS코드로 변환시키는데 그 변환시킨 JS코드가 createElement이고 그 createElement를 사용하면 자바스크립트 객체인 React Element가 만들어진다.

출처 : 처음 만난 리액트-인프런 강좌

아래는 React.createElement의 구조인데

type에는 문자열로된 html태그나 또다른 React Component가 들어간다.

모든 React Component는 최종적으로는 html태그를 사용하게된다.(하나의 컴포넌트는 여러개의 자식 컴포넌트로 구성되며 그 자식 컴포넌트를 분해하면 결국 html태그가 남게 된다)

 

props는 일단 간단하게 속성이라고 생각하면 되는데 태그에 class나 style같은 속성을 attribute라고 하는데 이 attribute보다 상위에 있는 복잡한 개념이고 일단 대충 속성이라는 것이라고 알고 넘어가자.

 

children은 해당 element의 자식 element가 여기 들어가게 된다.

출처 : 처음 만난 리액트-인프런 강좌

 

아래에는 Button 컴포넌트와 ConfirmDialog 컴포넌트가 있는데

ConfirmDialog 컴포넌트가 Button 컴포넌트를 포함하고 있다

출처 : 처음 만난 리액트-인프런 강좌

이런 ConfirmDiag 컴포넌트의 element의 형태는 어떻게 될까?

바로 다음과 같이 된다. 

 

1번째 children은 p태그이고

2번째 children은 리액트 component인 Button component이며

confirmdiag컴포넌트 element는 button컴포넌트의 element를 만들어서 합치게 된다

출처 : 처음 만난 리액트-인프런 강좌

그래서 최종적으로 confirmDiag 컴포넌트의 element의 형태는 다음과 같다.

출처 : 처음 만난 리액트-인프런 강좌


3. React elements의 특징

1) 불변성(immutable)

Element가 변하지 않는다면 화면이 어떻게 업데이트되는 것인가 궁금할 것이다.

출처 : 처음 만난 리액트-인프런 강좌

 

아래는 붕어빵을 예로 든 것이다.

붕어빵 틀로 붕어빵을 만드는데 안에 있는 붕어빵 앙코(팥소)를 바꿀 수 없는 것과 같은 이치이다.

앙코를 바꿀 순 없으니 새로 붕어빵을 만들어서 바꿔치기 하면 된다.

 

출처 : 처음 만난 리액트-인프런 강좌

Virtual DOM의 개념을 다시한번 상기시키면 된다.

바뀐 Element를 업데이트한다.

출처 : 처음 만난 리액트-인프런 강좌

리액트로 개발을 하다보면 상태관리와 더불어 화면이 얼마나 자주바뀌는지가 성능에 큰 영향을 미친다


4. Elements 렌더링하기

element를 만든 이후 화면에 보여주기 위해서는 렌더링이라는 과정을 거쳐야 한다

 

먼저 아래 코드를 보자.

아래 코드는 id가 root인 div태그이다.

아주 간단한 코드이지만 모든 리액트 앱에 들어가는 아주 중요한 코드이다.

 

실제로 이 div 태그안에 react element가 렌더링되며 이것을 root DOM node라고 한다.

div태그에 들어있는 모든 것이 react element에 의해 관리된다. 

 

출처 : 처음 만난 리액트-인프런 강좌

오직 리액트만으로 만들어진 모든 웹사이트들은 단 하나의 root DOM node를 가지며

기존에 있는 웹사이트에 리액트를 추가적으로 연동하게 되면 여러개의 분리된 수많은 root DOM node를 가질 수 있다

 

아래와 같이 최상단에 있는 node가 Root DOM Node이다

출처 : 처음 만난 리액트-인프런 강좌
출처 : 처음 만난 리액트-인프런 강좌

위 같은 div태그에react element를 렌더링하기 위해서는 다음과 같은 코드를 사용한다

 

element를 먼저 생성하고 생성된 element를 root div태그에 렌더링하는 코드이다.

이때 사용되는  render라는 함수는 render(a , b )의 형태에서

react elements인 a를 

html elements(Dom elements)인 b에 렌더링하는 역할을 한다.

 

오늘의 핵심은 다음과 같다.

element는 한번 생성되면 바뀔 수 없으므로 업데이트를 하기 위해서는 다시 생성해야한다.


참고문헌

1. [무료] 처음 만난 리액트(React) - 인프런 | 강의 (inflearn.com)

728x90
LIST

+ Recent posts