0.0 파트 0 (클릭 가능)
좋습니다! ReactDOM.render 부터 시작해보도록 합시다.
시작점은 ReactDom.render입니다. 앱은 여기서부터 DOM 렌더링을 시작합니다. 디버깅을 쉽게 하기 위해 간단한 컴포넌트 <ExampleApplication/>
를 만들었습니다. 첫번째로 발생되는 작업은 JSX가 리엑트의 엘리먼트로 변형되는 겁니다. 이건 매우 단순하며, 간단한 구조를 한 plain objects 입니다. 단지 컴포넌트의 render 메소드로부터 반환된 것을 나타낼 뿐입니다. 여기에 있는 props, key, ref 와 같은 몇 가지 필드들은 알고 계셔야 합니다. type 프로퍼티는 JSX에서 설명하는 마크업 객체를 참조합니다. 예제의 경우, ExampleApplication
클래스 일 수도 있지만, 버튼 태그에 대한 문자열 button
일 수도 있습니다. 또한 리엑트 엘리먼트를 생성하는 동안 리엑트는 defaultProps
를 props
(지정된 경우)와 머지하고 propTypes
을 확인합니다.
더 자세한 내용은 소스 코드를 확인하시기 바랍니다.: src\isomorphic\classic\element\ReactElement.js
ReactMount
(01)라는 모듈이 보이실겁니다. 이건 컴포넌트 마운팅의 로직을 포함하고 있습니다. 실제로, ReactDOM
에는 로직이 없습니다. 이건 단지 ReactMount
를 사용하기위한 인터페이스일 뿐입니다. 따라서 ReactDOM.render
를 호출하면 기술적으로 ReactMount.render
를 호출합니다. 그렇다면 마운팅이란건 무엇일까요?
마운팅이란 대표적인 DOM 엘리먼트를 생성하고 제공된
container
에 삽입하여 리엑트 컴포넌트를 초기화하는 작업.
최소한 코드의 주석은 그렇게 설명하고 있습니다. 이것의 진짜 의미는 무엇일까요? 다음의 변환을 상상해 보도록 하죠:
0.1 JSX에서 HTML로 (클릭 가능)
리엑트를 document에 넣기위해서 컴포넌트 명세를 HTML로 변환하는 것이 필요합니다. 그럼 그 명세를 어떻게 전달할 수 있을까요? 맞습니다, 우선 모든 props, 이벤트 리스너, 중첩 컴포넌트 로직들을 다뤄야 필요가 있습니다. 이건 높은 수준의 명세(컴포넌트)를 웹 페이지에 넣을 수 있는 낮은 수준의 데이터(HTML)로 세분화 합니다. 이게 마운팅에 대한 전부입니다.
0.2 JSX에서 HTML, 확장 (클릭 가능)
좋아요, 계속 해보도록 합시다. 그런데... 잠깐 재미있는 사실을 좀 알아볼까요?
흥미로운 사실 : 스크롤이 모니터링 중인지 확인해보세요.(02)
재밌는 점은 root 컴포넌트를 처음 렌더링하는 동안, 리엑트는 스크롤 리스너 초기화하고 스크롤 값을 캐시를 해서, 리플로우(reflows) 발생 없이 응용프로그램 코드가 그것들에게 접근할 수 있습니다. 사실, 브라우저마다 렌더링 구현이 다르게 때문에, 어떤 DOM은 정적이지 않고 코드에서 사용할 때마다 계산됩니다. 물론, 이건 성능에 영향을 미칩니다. 이것은
pageX
및pageY
를 지원하지 않는 예전 브라우저에 해당됩니다. 리엑트는 이것 또한 최적화 하려고 하고있습니다. 보시다시피, 빠른 도구를 만든다는 것에는 많은 기술들이 사용되는데, 스크롤이 좋은 예 입니다.
위의 스키마에 숫자(03)으로 표현된 '인스턴스 생성'이 보이지만, 아직 <ExampleApplication />
의 인스턴스를 만드는 것은 이른것 같습니다. 사실, TopLevelWrapper
(내부 React 클래스)를 인스턴스화합니다. 이제 다음 스키마를 확인해 봅시다.
0.3 JSX에서 VDOM으로 (클릭 가능)
세단계를 확인하실 수 있는데, 리엑트 엘리먼트를 통한 JSX는 내부 리엑트 컴포넌트 유형 중 하나로 변환됩니다. : ReactCompositeComponent
(자체 컴포넌트 용), ReactDOMComponent
(HTML 태그 용), ReactDOMTextComponent
(text nodes 용). ReactDOMTextComponent
를 생략하고 처음 두 개만 집중하도록 하겠습니다.
내부 컴포넌트란 뭘까요? 음, 이건 좀 흥미롭습니다. Virtual DOM에 대해 들어 보신적 있을겁니다. Virtual DOM은 리엑트가 diff 연산중에 DOM을 직접 건드리지 않는데 사용되는 DOM 표현의 일종입니다. 이것이 리엑트를 빠르게 하죠. 그러나 실제로 리엑트 소스코드에는 'Virtual DOM'이라는 파일이나 클래스가 없습니다. 재미있지 않나요? Virtual DOM은 단지 개념일 뿐이며, 실제 DOM을 조작하기 위한 접근방식 입니다. 그래서 몇몇 사람들은 Virtual DOM 항목이 리엑트 엘리먼트를 가리킨다고 말하지만, 그건 정확하게는 맞지 않다고 생각됩니다. 저는 Virtual DOM은 ReactCompositeComponent
, ReactDOMComponent
, ReactDOMTextComponent
라는 세 클래스를 가리킨다고 생각합니다. 왜 그런지는 나중에 확인해보도록 하죠.
좋습니다, 이제 인스턴스 만드는 작업을 끝내보도록 합시다. ReactCompositeComponent
의 인스턴스를 만들것이지만, 지금은 아닙니다. 실제로는 <ExampleApplication/>
을 ReactDOM.render
에 넣었기 때문이죠. 리엑트는 항상 TopLevelWrapper
에서 컴포넌트 트리를 렌더링하기 시작합니다. 이건 거의 비어있는 래퍼이며, render
(컴포넌트의 렌더링 메서드) 나중에 <ExampleApplication />
을 반환합니다.
// src\renderers\dom\client\ReactMount.js #277
TopLevelWrapper.prototype.render = function () {
return this.props.child;
};
그래서, 지금은 오직TopLevelWrapper
만 생성되었습니다.
더 해보도록 하죠. 하지만... 우선, 또 흥미로운 사실을 알아볼까요?
흥미로운 사실 : DOM 중첩 유효성 검사
중첩된 컴포넌트들이 렌더링 될 때마다
validateDOMNesting
이라는 HTML 유효성 검사를 위한 전용 모듈에 의해 유효성이 검사됩니다. DOM 중첩 유효성 검사는child -> parent
태그 계층의 검증을 의미합니다. 예를 들어 부모 태그가<select>
이면 자식 태그는option
,optgroup
,#text
중 하나만 있어야합니다. 이 규칙은 실제로 https://html.spec.whatwg.org/multipage/syntax.html#parsing-main-inselect에 정의되어 있습니다. 이미 다음과 같은 에러를 내는 모듈을 봤을겁니다. : <div> cannot appear as a descendant of <p> .
우리가 어떻게 여기까지 왔는지 다시 한번 살펴보도록 합시다. 스키마에서 덜 중요한 부분을 제거하면 다음과 같습니다.
0.4 간단히 보는 파트 0 (클릭 가능)
공백제거와 정렬을 통해 보기 좋게 수정했습니다.
0.5 간단히 보는 파트 0 리펙토링 버전 (클릭 가능)
좋습니다. 사실, 이것이 여기서 일어나는 일 전부입니다. 이제 파트 0의 핵심들을 가지고 최종 mounting
스키마에 사용할 수 있습니다.
0.6 파트 0의 핵심 (클릭 가능)
우리는 해냈습니다!