JavaScript engine fundamentals: optimizing prototypes

이 포스트는 JavaScript engine fundamentals: optimizing prototypes 의 글을 번역한 것입니다.

이 문서는 V8 엔진뿐만 아니라 모든 JavaScript 엔진에 공통으로 적용되는 몇 가지 핵심 기본 사항을 설명합니다. JavaScript 개발자로서, JavaScript 엔진이 어떻게 작동하는지에 대한 이해를 통해 코드의 성능 특성을 추론할 수 있을것입니다.

이전 포스트에서, 우리는 JavaScript 엔진이 Shapes와 Inline Caches를 사용하여 어떻게 object와 array에 대한 접근을 최적화하는지에 대해서 논의했습니다. 이 글은 optimization pipeline의 trade-off에 관해서 설명하고, JavaScript 엔진이 prototype property에 대한 접근 속도를 높이는 방법을 설명합니다.

Note : 문서를 읽는 것보다 발표를 보는 것을 더 좋아한다면 아래의 동영상을 보십시오. 그렇지 않다면 문서를 계속해서 읽어 주십시오.

Optimization tiers and execution trade-offs

이전 포스트에서, 현대 JavaScript 엔진들이 전반적으로 동일한 pipeline(아래의 그림과 같이)을 가지는 방법에 대해서 이야기를 나눴습니다.

0
js engine pipeline

또한 엔진 간에 high-level의 pipeline은 유사하지만 optimization pipeline에는 종종 차이가 있음을 지적했습니다. 왜 그런 것일까요? 일부 엔진의 최적화 tier가 다른 엔진보다 많은 이유는 무엇일까요? 그것은 아래의 그림과 같이, 코드를 빨리 만들어서 실행하는 것과 시간을 더 들이지만 결국 최적화된 성능으로 코드를 만들어서 실행하는 것 사이의 trade-off가 있기 때문입니다.

1
trade-off startup speed

Interpreter는 bytecode를 재빨리 만들 수 있지만, 일반적으로 bytecode는 그리 효율적이지 않습니다. 반면에 optimizing compiler는 시간이 좀 더 걸리지만 결국 훨씬 더 효율적인 machine 코드를 생산합니다.

아래의 그림이 바로 V8이 사용하는 model입니다. V8 interpreter는 Ignition이라고 불리며, 모든 JavaScript 엔진의 interpreter 중에서 가장 빠릅니다(raw bytecode 실행 측면에서) V8의 optimizing compiler이름은 TurboFan이며, 고도로 최적화된 machine 코드를 생성합니다.

2
trade-off startup speed: V8

이러한 startup latency와 execution speed 간의 trade-off는 일부 JavaScript 엔진이 최적화 tier를 추가하기로 선택하는 이유입니다. 예를 들어 SpiderMonkey는 interpreter와 전체 IonMonkey 사이에 compiler를 최적화하는 Baseline tier를 추가했습니다.

3
trade-off startup speed: SpiderMonkey

Interpreter는 bytecode를 빠르게 생성하지만, bytecode의 실행속도는 비교적 느린편입니다. Baseline은 코드를 생성하는데 약간 더 오래 걸리지만 run-time 성능을 향상시킵니다. 마지막으로, optimizing compiler인 IonMonkey는 machine 코드를 생산하는데 가장 오래 걸리지만, machine 코드는 매우 효율적으로 실행됩니다.

구체적인 예시를 통해 각 JavaScript 엔진의 파이프라인이 어떤 방법으로 처리하는지 살펴보겠습니다. 여기 자주 반복되는 코드들이 있습니다.

let result = 0;
for (let i = 0; i < 4242424242; ++i) {
	result += i;
}
console.log(result);

V8은 Ignition interpreter에서 bytecode를 실행하기 시작합니다. 어느 시점에 V8은 코드가 hot 하다고 판단하고 TurboFan frontend를 시작합니다. TurboFan frontend는 profiling data를 통합하고 코드의 기본적인 machine representation 체계를 구축하는 TurboFan의 일부입니다. 그런 다음 추가적인 개선을 위해 다른 thread에 있는 TurboFan optimizer로 전송합니다.

4
pipeline detail: V8

Optimizer가 실행되는 동안, V8은 Ingition에서 bytecode를 실행합니다. 어느 시점에서 optimizer가 완료되고 나면 실행 가능한 machine 코드가 생성되며, 이 machine 코드로 계속해서 실행할 수 있습니다.

SpiderMonkey engine 역시 interpreter에서 bytecode를 실행하기 시작합니다. 그러나 Baseline tier가 추가되어 hot 코드는 먼저 Baseline으로 전송됩니다. Baseline compiler는 메인 thread에서 Baseline 코드를 생성하고 준비가되면 실행을 계속합니다.

5
pipeline detail: SpiderMonkey

한동안 Baseline 코드를 실행하면 SpiderMonkey는 결국 IonMonkey frontend를 실행하고 V8과 매우 유사한 optimizer를 시작합니다. IonMonkey가 최적화하는 동안 Baseline에서 실행됩니다. 마지막으로 optimizer가 완료되면, Baseline 코드 대신 최적화된 코드를 실행합니다.

Chakra의 구조는 SpiderMonkey와 매우 비슷하지만, Chakra는 메인 thread를 막지 않기 위해 더 많은 것들을 동시에 실행하려고 시도합니다. 메인 Thread에서 compiler의 일부를 실행하는 대신, Chakra는 compiler가 필요로 하는 bytecode와 profiling data를 복사하여 전용 compiler 프로세스로 보냅니다.

6
pipeline detail: Chakra

생성된 코드가 준비되면, Chakra 엔진은 bytecode 대신 이 SimpleJIT 코드를 실행합니다. FullJIT에 대해서도 같은 과정이 실행됩니다. 이러한 접근 방식의 이점은 복사가 발생하는 일시 중지 시간이 보통 전체 compiler가(frontend)를 실행하는 데 비해 훨씬 짧다는 것입니다. 그러나 이러한 접근 방식의 단점은 copy heuristic 이 특정 최적화에 필요한 일부 정보를 놓칠 수 있다는 것입니다. 따라서 latency를 위해 코드 품질을 어느 정도 타협한 것입니다.

JavaScriptCore에서 모든 optimizing compiler는 모두 main JavaScript 실행과 동시에 일어나기 때문에 copy 단계가 없습니다! 대신에 메인 thread는 다른 thread의 compile 작업만 일으킵니다. 그 후 해당 compiler는 복잡한 locking scheme을 사용해서 메인 thread에서 profiling data에 접근합니다.

7
pipeline detail: JavaScriptCore

이 접근법의 장점은, 메인 thread에서 일어나는 JavaScript 최적화로 인한 junk를 감소시킨다는 것입니다. 단점은 복잡한 multithreading 문제를 처리해야 하는 것과 다양한 작업에 대해 약간의 locking cost를 감수해야 한다는 것입니다.

우리는 interpreter와 같이 빨리 코드를 생성하는 방법과 optimizing compiler로 빠른 코드를 생성하는 방법 사이의 trade-off에 관해서 이야기했습니다. 하지만 여기에는 또 다른 trade-off가 있습니다. 바로 memory usage(메모리 사용량) 입니다. 이를 설명하기 위해, 여기 두 개의 수를 더하는 간단한 JavaScript 프로그램이 있습니다.

function add(x, y) {
	return x + y;
}

add(1, 2);

아래의 코드는 위의 add function을 V8의 interpreter인 Ignition을 사용해서 만든 bytecode입니다.

StackCheck
Ldar a1
Add a0, [0]
Return

위의 bytecode에 대해서 걱정하지 마십시오. 실제로 읽을 필요는 없습니다. 요점은 단지 4개의 명령문으로 이루어져 있다는 것입니다.

코드가 hot해지면, TurboFan은 아래와 같이 고도로 최적화된 machine 코드를 생성합니다.

leaq rcx,[rip+0x0]
movq rcx,[rcx-0x37]
testb [rcx+0xf],0x1
jnz CompileLazyDeoptimizedCode
push rbp
movq rbp,rsp
push rsi
push rdi
cmpq rsp,[r13+0xe88]
jna StackOverflow
movq rax,[rbp+0x18]
test al,0x1
jnz Deoptimize
movq rbx,[rbp+0x10]
testb rbx,0x1
jnz Deoptimize
movq rdx,rbx
shrq rdx, 32
movq rcx,rax
shrq rcx, 32
addl rdx,rcx
jo Deoptimize
shlq rdx, 32
movq rax,rdx
movq rsp,rbp
pop rbp
ret 0x18

코드가 매우 많습니다. 특히 4개의 명령문으로 이루어진 bytecode와 비교하면 더 그렇습니다. 보통 bytecode는 machine 코드보다, 특히 최적화된 machine 코드보다 더 간결한 경향이 있습니다. 반면에, bytecode는 실행하는데 interpreter가 필요하지만, 최적화된 machine 코드 processor에 의해 바로 실행될 수 있습니다.

이것이 바로 JavaScript 엔진이 “모든 것을 최적화”하지 않는 주된 이유 중 하나입니다. 우리가 앞서 살펴봤듯이, 최적화된 machine 코드를 생성하는데는 긴 시간이 걸리고, 그 외에도 최적화된 machine 코드는 더 많은 memory가 필요하다는 것을 알게 되었습니다.

8
trade-off memory

Summary: JavaScript 엔진들의 최적화 tier가 다른 이유는 interpreter와 같이 코드를 빠르게 생성하는 것과 최적화 compiler로 빠른 코드를 생성하는 것 사이의 근본적인 trade-off 때문입니다. 최적화 계층을 추가하면 복잡성과 오버헤드를 추가로 희생하여 보다 fine-grained한 의사 결정을 내릴 수 있습니다. 또한, 생성된 코드의 memory-usage와 최적화 level 간의 trade-off도 있습니다.(machine 코드와 bytecode) 그렇기 때문에 JavaScript 엔진은 hot한 function만 최적화하려고 합니다.

Optimizing prototype property access

우리의 이전 글에서 JavaScript엔진이 Shape와 Inline cache를 사용해서 어떻게 object property load를 최적화 했는지 설명하였습니다. 다시 상기해보면, 엔진은 object의 값과 별도로 object의 Shape를 저장합니다.

9
shape

Shape는 Inline Caches 혹은 ICs 라 불리는 최적화를 가능하게 합니다. Shape 및 ICs를 코드의 동일한 위치에서 property에 반복해서 접근할 때 속도를 높일 수 있습니다.

10
Inline Cache

Classes and prototype-based programming

이제 JavaScript object의 property를 빨리 가져오는 방법에 대해서 알았으니, JavaScript에서 최근에 추가된 Class에 대해서 살펴보겠습니다. 아래의 코드는 JavaScript의 class syntax입니다.

class Bar {
	constructor(x) {
		this.x = x;
	}
	getX() {
		return this.x;
	}
}

마치 JavaScript에서 새로운 개념이 나온것처럼 보이지만, 단지 JavaScript에서 계속 사용한 prototype-based programming을 위한 syntactic sugar일 뿐입니다.

function Bar(x) {
	this.x = x;
}

Bar.prototype.getX = function getX() {
	return this.x;
};

여기 Bar.prototype object에 getX property를 할당합니다. 이 작업은 다른 어떤 object에서도 똑같이 일어납니다. JavaScript에서 prototype은 object일 뿐이기 때문입니다. JavaScript와 같은 prototype-based programming language에서는, method는 prototype을 통해 공유되고 field는 실제 instance에 저장됩니다.

Bar의 새로운 instance인 foo가 생성될 때 뒤에선 어떤 일이 일어나는지 살펴보겠습니다.

const foo = new Bar(true);

위의 코드를 통해서 생성된 instance는 'x' property 가 들어있는 shape를 가진다. foo의 prototype인 Bar.prototypeBar class에 속해있다.

11
class shape 1

Bar.prototype은 단일 property 'getX'를 포함하는 shape를 가지고 있으며, getX의 값은 호출시 this.x를 반환하는 함수입니다. Bar.prototype의 prototype은 Object.prototype이고, 이는 JavaScript 언어의 한 부분입니다. Object.prototype은 prototype tree의 root이기 때문에 이것의 prototype은 null입니다.

12
class shape 2

만약 같은 class의 다른 instance를 만든다면, 두 instance는 앞서 살펴본것과 같이 object shape를 공유합니다. 그리고 두 instance 모두 같은 Bar.prototype object를 가리킬것입니다.

Prototype property access

자, 이제 class를 선언하고 새로운 instance를 생성할 때 어떤 일이 일어나는지 알게 되었습니다. 하지만, 아래 코드와 같이 instance의 method를 호출할 때는 어떻게 될까요?

class Bar {
	constructor(x) { this.x = x; }
	getX() { return this.x; }
}

const foo = new Bar(true);
const x = foo.getX();
//        ^^^^^^^^^^

모든 method 호출을 두 가지 개별 단계로 생각할 수 있습니다.

const x = foo.getX();

// is actually two steps:

const $getX = foo.getX;
const x = $getX.call(foo);

첫 번째 단계는 prototype의 property일 뿐인 method를 가져오는 것입니다(이러한 값이 function일 수 있습니다). 두 번째 단계는 instance를 this로 사용해서 function을 call하는 것입니다. Instance foo에서 method getX를 가져오는 첫 번째 단계를 살펴보겠습니다.

13
method load

Engine은 foo instance에서 시작하고, foo의 shape에 'getX’ property가 없다는 것을 알아챕니다. 그래서 engine은 prototype chain을 타고 올라간다. Bar.prototype에 도달하여, prototype의 shape를 봤을 때 offset 0'getX' property가 있는 것을 알 수 있습니다. Bar.prototype에서 앞서 찾은 offset의 값을 보면 우리가 찾던 JSFunction getX를 찾을 수 있습니다.

JavaScript의 유연함(flexibility)덕분에 prototype chain link를 다음과 같이 변형할 수 있습니다.

const foo = new Bar(true);
foo.getX();
// → true

Object.setPrototypeOf(foo, null);
foo.getX();
// → Uncaught TypeError: foo.getX is not a function

위의 예제에서 foo.getX를 2번 호출했지만, 각각의 호출은 완전히 다른 의미와 결과를 가져옵니다. 그렇기 때문에 prototype은 JavaScript에서 object일 뿐이지만, 일반 object의 자체(own) property에 접근하는 속도를 높이는 것 보다 prototype의 property에 접근하는 속도를 높이는 것이 더 어렵습니다.

일반적인 프로그램들을 보면, prototype의 property를 로드하는 작업은 매우 빈번하게 일어납니다.(method를 호출할때마다 말이죠.)

class Bar {
	constructor(x) { this.x = x; }
	getX() { return this.x; }
}

const foo = new Bar(true);
const x = foo.getX();
//        ^^^^^^^^^^

앞서 우리는 어떻게 엔진이 Shape와 Inline Cache를 사용하여 정기적인 자신의 property 로드를 최적화하는지 논의했습니다. 유사한 shape의 object에서 prototype property의 반복적 로드를 어떻게 최적화할 수 있을까요? 우리는 위에서 property 로드가 어떻게 일어나는지 보았습니다.

14
property load check 1

위의 그림과 같은 경우, 반복되는 로드에 대해 신속하게 처리하려면 다음 세 가지 사항을 알아야 합니다.

  1. foo의 shape는 'getX'를 포함하지 않으며, 변하지 않습니다. 즉, foo의 property를 추가하거나 삭제하지않고, 그리고 property attribute중 하나라도 변경하지 않는다는 뜻입니다.
  2. foo의 prototype은 초기의 Bar.prototype입니다. 즉, Object.setPrototypeOf()를 사용하거나 특별한 _proto_ property에 값을 할당함으로써 foo의 prototype을 변경하지 않는다는 뜻입니다.
  3. Bar.prototype의 shape은 'getX'를 가지며 변하지 않습니다. 즉, Bar.prototype의 property를 추가하거나 삭제하지않고, 그리고 prototype attribute중 하나라도 변경하지 않는다는 뜻입니다.

일반적인 경우, instance 자체에 대해 1회의 검사(1번)를 수행하고, prototype에 대해 2회의 검사(2, 3번)를 수행해야 하며, 이 검사에서는 우리가 찾고 있는 prototype이 포함된 prototype까지 올라갑니다. 1+2N체크(여기서 N은 관련된 prototype의 수임)는 prototype 체인이 상대적으로 얕기 때문에 이 경우 나쁘지 않게 들릴 수 있지만, 엔진은 흔히 일반적인 DOM class 훨씬 더 긴 prototype chain을 처리해야 합니다. 그 예는 다음과 같습니다.

const anchor = document.createElement('a');
// → HTMLAnchorElement

const title = anchor.getAttribute('title');

우리는 HTMLAnchorElement를 가지고 있으며 여기서 getAttribute() 함수를 호출하였습니다. 이 단순한 anchor element의 prototype chain은 이미 6개의 prototype이 포함되어 있습니다. 대부분의 흥미로운 DOM method의 대부분은 HTMLAnchorElement prototype이 아니라 chain의 더 높은곳에 있습니다.

15
anchor prototype chain

getAttribute() 함수는 Element.prototype에 있습니다. 즉, anchor.getAttribute()를 호출할 때마다 JavaScript는…

  1. 'getAttribute'anchor object 자체에 없는지 확인하고,
  2. 바로 위의 prototype이 HTMLAnchorElement.prototype인지 확인하고,
  3. 'getAttribute'가 없다는것을 확인하고,
  4. 다음 prototype이 HTMLElement.prototype이라는 것을 확인하고,
  5. 여기도 'getAttribute'가 없다는것을 확인하고,
  6. 결국 다음 prototype이 Element.prototype이라는 것을 확인하고,
  7. 그리고 거기에 'getAttribute'가 있다.

총 7번의 확인을 해야합니다! 이러한 종류의 코드는 웹에서 매우 흔하기 때문에, 엔진은 prototype의 property 로드에 필요한 검사 횟수를 줄이기 위해 트릭을 사용합니다.

앞에서 살펴봤던 예제로 돌아가서, 우리는 foo'getX'에 접근하기 위해서 3번의 확인을 거칩니다.

class Bar {
	constructor(x) { this.x = x; }
	getX() { return this.x; }
}

const foo = new Bar(true);
const $getX = foo.getX;

이 property를 가지는 prototype이 나올 때까지 각 object에 대해서 shape 검사를 해야 합니다. 만약 prototype 검사를 property 검사에 넣어서 검사 횟수를 줄일 수 있다면 좋을 것 같습니다. 그리고 이것이 JavaScripe 엔진이 기본적으로, 그리고 간단하게 사용하는 트릭입니다. Prototype link를 instance 자체에 저장하는 대신, Shape에 저장하는 것입니다.

16
prototype load cheks 2

각 shape는 prototype을 가리킵니다. 이것은 foo의 prototype이 변경될 때마다 shape도 변경되는 것을 뜻합니다. 이제 우리는 특정 property의 부재와 prototype link를 보기 위해서 object의 shape만 확인하면 가능하게 되었습니다.

이 접근 방식을 사용하면 prototype에 대한 property access 속도를 높이기 위해 필요한 검사 횟수를 1+2N에서 1+N으로 줄일 수 있습니다. 하지만 해당 비용은 prototype chain 길이만큼 선형적으로 늘어나기 때문에 여전히 꽤 많은 비용이 들어가는 편입니다. 엔진은 검사 횟수를 상수가 되도록 더 줄이기 위해 다양한 트릭을 구현합니다(특히 동일한 property 로드의 후속 실행에 대해).

Validity cells

V8은 이러한 목적을 위해 특별하게 prototype shape를 처리합니다. 각 prototype은 다른 object(특히 다른 prototype과는 공유되지 않음)와 공유되지 않는 고유한 shape를 가지며, 각 prototype shape에는 특별한 ValidityCell이 연결됩니다.

17
validity cell

이 ValidityCell은 관련된 prototype 또는 위에 있는 prototype을 변경할 때마다 무효화됩니다. 이것이 정확히 어떻게 작동하는지 살펴봅시다.

특정 prototype에서 후속(subsequent) 로드의 속도를 높이기 위해서 V8은 4개의 field를 가지는 Inline Cache를 배치합니다.

18
inline cache - validity cell

이 코드를 처음 실행하는 동안 inline cache를 워밍업할 때 V8은 해당 property가 발견된 offset, 즉 해당 property가 발견된 prototype(이 예제에서는 Bar.prototype)과 instance의 모양(여기서는 foo의 shape)을 기억합니다. 그리고 instance의 shape가 가리키는 바로위의 prototype(이 경우 Bar.prototype)의 ValidityCell을 가리키는 링크도 가지고 있습니다.

다음번에 Inline Cache를 hit하면, 엔진은 instance의 shape와 ValidityCell을 확인합니다. 아직 valid하다면, 엔진은 Prototypeoffset에 직접 접근해서 추가적인 조회를 생략할수 있습니다.

19
validity cell - Invalid

Prototype이 변경되면 새로운 shape가 prototype에 할당되고 기존의 ValidityCell 무효(invalidated)가 됩니다. 따라서 다음 번 실행시에 Inline Cache는 누락되어서 성능이 저하됩니다.

이전에 살펴봤던 DOM element예제로 돌아가보면, 예제의 Object.prototype을 변경하는 것은 Object.prototype의 Inline Cache가 무효화되는것 뿐만 아니라, 아래에 있는 EventTarget.prototype, Node.prototype, Element.prototype 그리고 마지막으로 HTMLAnchorElement.prototype사이에 있는 prototype 모두 무효화되는 것을 뜻한다.

20
prototype chain - validity cell

실제로 코드가 실행되는 동안 Object.prototype을 수정한다는 것은 성능따위 창문밖으로 던져버린다는 것을 뜻합니다. 절대 하지마세요!

구체적인 예를 들어 좀 더 알아보겠습니다. Bar라는 class가 있다고 가정하고, Bar object에 대한 method를 호출하는 함수 loadX가 있다고 해보겠습니다. 동일한 class의 instance에서 이 loadX 함수를 몇 번 호출해보겠습니다.

class Bar { /* … */ }

function loadX(bar) {
	return bar.getX(); // IC for 'getX' on `Bar` instances.
}

loadX(new Bar(true));
loadX(new Bar(false));
// IC in `loadX` now links the `ValidityCell` for
// `Bar.prototype`.

Object.prototype.newMethod = y => y;
// The `ValidityCell` in the `loadX` IC is invalid
// now, because `Object.prototype` changed.

loadX에 있는 inline cache는 Bar.prototypeValidityCell을 가리킵니다. 만약 Object.prototype(JavaScript의 prototype들 중에서 가장 root에 있음)에 어떤 변형을 주는 작업을 한다면, 그 ValidityCell은 무효화되고 다음번 실행시에 기존의 Inline Cache가 누락되고 성능이 저하됩니다.

Object.prototype를 변형시키는 것은 언제나 좋지않은 아이디어입니다. 왜냐하면 엔진이 그 시점까지 prototype 로드하면서 생긴 모든 Inline Cache를 무효화시키기 때문입니다. 다음 코드는 하지 말아야 할 다은 예입니다.

Object.prototype.foo = function() { /* … */ };

// Run critical code:
someObject.foo();
// End of critical code.

delete Object.prototype.foo;

위의 코드에서 Object.prototype를 확장했고, 이는 그 시점까지 엔진이 올려놓은 모든 prototype의 Inline Cache를 무효로 함을 뜻합니다. 그런 다음 새로 만든 prototype method를 사용하는 코드를 실행합니다. 엔진은 처음부터 다시 시작해서, 모든 prototype property에 대한 접근을 위해 새로운 Inline Cache를 만들어야 합니다. 그리고 마지막으로 “스스로 깨끗하게 정리(clean up after ourselves)”하기 위해서 앞서 추가했던 prototye method를 삭제합니다.

정리란 좋은 아이디어처럼 들립니다. 그렇지 않나요? 글쎄요, 위의 경우에서는 좋지 않은 상황을 더욱 악화시키는 요소로 작용합니다. Property의 삭제는 Object.prototype의 수정을 뜻하므로 Inline Cache는 모두 다시 무효가 되고 엔진은 다시 처음부터 다시 작업을 해야 합니다.

Summary: Prototype은 object일 뿐이지만 prototype에서 method 검색 성능을 최적화하기 위해 JavaScript 엔진에서 특별하게 처리됩니다. Prototype은 건드리지 마십시오! 또는 prototype을 정말 건드려야 하는 경우, 다른 코드가 실행되기 전에 prototype을 제작하십시오. 그러면 당신의 코드가 실행되는 동안 엔진의 모든 최적화가 무효화되지 않습니다.

Take-aways

JavaScript 엔진이 object와 array를 저장하는 방법과, Shape, Inlice Cache 그리고 ValidityCell을 통해 prototype 작업을 최적화하는 방법을 배웠습니다. 이러한 지식을 바탕으로 성능을 향상시킬 수 있는 몇 가지 실용적인 JavaScript 코딩 팁을 확인했습니다. prototype을 건드리지 마십시오(만일 정말로, 정말로 필요할때는 다른 코드들을 실행하기 전에 건드리십시오).