Grasp “By Value” and “By Reference” in JavaScript - Lee-hyuna/33-js-concepts-kr GitHub Wiki
자바스크립트는 객체 형태를 지향하는 언어이다. 이 뜻은 자바스크립트의 대부분이 객체로 이루어져있다는 것을 의미한다.
예를들어 function은 object이다. 객체가 아닌 유일한 요소는 기본 데이터 유형이다 (string, number, boolean, null, undefined).
기본데이터 유형은 불변성을 가지고 있어 생성되면 수정할 수 없다.
둘 사이의 차이점 중 하나는 원시 데이터 유형은 값으로 전달되고 개체는 참조로 전달된다는 것이다.
이게 무슨 의미인지는 다음과 같이 생각할 수 있다.
Value는 원본을 복사
한 것을 생성하는 것을 의미한다.
두명의 쌍둥이가 있다고 가정해보자: 그들은 정확히 똑같이 태어났지만, 첫 째가 전쟁에서 다리를 잃었다고 둘째 까지 다리를 잃지는 않는다.
원본에 대한 가명(별명)
이 참조에 의해 생성
예를 들어 엄마가 나를 부를때 'Pumpkin Pie'라고 부를 때, 내 이름은 Margaret이지만, 갑자기 'Pumpkin Pie'라는 복제품이 생기지는 않는다. 즉, 비슷한 의미로 나는 한 사람이지만 여러개의 이름으로 불릴 수 있다.
개체의 동작 방식을 먼저 할당 연산자(=)와 함께 값을 할당하고, 두 번째 매개변수로서 함수에 전달했을 때 검토해보자.
초기 데이터 유형의 경우 =로 할당되는 방법
var name = "Carlos";
var firstName = name;
name = "Carla";
console.log(name); // "Carla"
console.log(firstName); // "Carlos"
이 결과는 너무 간단하게도 일치하는 연산자이다. 여기서 실제로 일어나는 일은 다음과 같이 단순화할 수 있다.
- 변수
name
이 생성 되었을 때 'Carlos'라는 값을 할당 받았다. JavaScript는 이를 위해 메모리 스팟을 할당한다. - 변수
firstName
이 생성이 되었을 때name
의 값을 _복사_하게 된다.firstName
은 자체적으로 메모리 스팟을 갖고 있고name
과는 독립되어있다. 이 시점에서firstName
은 'Carlos'라는 값을 갖게 된다. - 그리고
name
의 값이 'Carla'로 바뀌고,firstName
은 여전히 원래의 값을 가지고 있는데 그 이유는 다른 메모리 스팟을 가지고 있기 때문이다.
원형으로 작업할 때 = operator는 원래 변수의 사본을 생성한다. 그게 가치관이야.
기본적으로 할당된 값을 갖고 작업할 때는 변수를 복사하여 생성하는 것이 좋다.
var myName = {
firstName: "Carlos"
};
var identity = myName;
myName.firstName = "Carla";
console.log(myName.firstName); // "Carla"
console.log(identity.firstName); // "Carla"
객체를 감싸고 있는 변수에 대한 output은 동일하게 작동한다. 값 할당에 대한 부분은 =가 동일하게 동작하기 때문이다
- 변수
myName
은 생성될 때 객체로 생성이 되고firstName
프로퍼티를 생성한다. 그 후firstName
은 'Carlos'라는 값을 갖게된다. 자바스크립트는 객체를 포함하고 있는myName
를 메모리스팟에 할당하게 된다. - 변수
identity
는 생성이 되고myName
을 가르키게 된다. 여기서 이 변수는identity
에 대한 메모리 공간이 없게 되고 오롯이myName
의 값만 바라보게 된다 -
myName
의 프로퍼티인firstName
의 값이 'Carla'로 변경이 되면서identity
의 값 또한 변경이 된다.
myName.firstName
를 기록할 때 새 값이 표기가 되고 이 부분은 매우 간단하다. 그러나 identity.firstName
을 기록할 때 myName.firstName
까지 새로운 'Carla'값잉 할당이 된다. 이 것은 identity.firstName
이 오로지 myName.firstName
의 메모리 장소를 따라가기 때문이다.
Object로 작업할 때 =로 할당되는 것은 원래의 객체에 대해 비슷한 객체를 생성하는 것이고 새로운 객채를 생성하지 않는다. "by reference"라는 말이 바로 그 뜻이다.
기본 데이터 유형이 값에 따라 함수에 전달됨
함수 내에서 초기 데이터 유형의 값을 변경하면 외부 범위의 변수에 영향을 주지 않음:
var myName = "Carlos";
function myNameIs(aName){
aName = "Carla";
}
myNameIs(myName);
console.log(myName); // "Carlos"
function이 호출된 후 myNameIs
내부에 있는 변수 myName
를 바꾸려고 할 때, 이 것은 여전히 'Carlos'라는 값을 가지고 있다.
그 이유는, 초기값 타입이 통과가 되면서 값 또한 통과가 되었기 때문이다.
myName의 사본을 전달하고 있다: myName 내부에서 수행하는 모든 작업은 myName
변수가 아닌 myName
복사본을 전달하기 때문에 전역 범위에서 myName
변수에 대해 영향을 주지 않는다.
객체가 참조로 함수에 전달됨
참고자료로 어떤 것을 전달하고 있을 때, 그 물체의 사본이 아닌 다른 것을 가리키는 것을 전달하고 있는 것이다. 따라서 자바스크립트가 참조로 객체를 통과하기 때문에 함수 내에서 해당 객체의 속성을 변경하면 변경 내용이 외부 범위에 반영된다.
var myName = {};
function myNameIs(aName){
aName.firstName = "Carla";
}
myNameIs(myName);
console.log(myName); // Object {firstName: "Carla"}
myNameIs
function이 호출이 되고 난 후 myName
변수는 firstName
의 key가 'Carla'라는 값을 가지게 된다. 이 값은 함수에 인자로 전달 했을 때 global scope에 의해 변화되었다
이 것은 함수에 전달되면서 사본을 전달하지 않기 때문이다. myName
객체에 대해 가르키고 있고 그 것을 전달하고 있다. 따라서 함수에서 그 속성을 변경할 때 외부 범위에서 속성을 바꾸는 것이다.
그러나 당신이 알아야 할 미묘한 점은 다음과 같다:
var myName = {
firstName: "Carla"
};
function myNameIs(aName){
aName = {
nickName: "Carlita"
};
}
myNameIs(myName);
console.log(myName); // Object {firstName: "Carla"}
외부 범위에서 변수 myName
의 값을 프린트 하고 객체에 닉네임 속성을 추가하지 않았다. 왜 그런 것일까?
자세히 살펴보면, 함수를 통해 하고 있는 것은 myName
객체에 새로운 값을 재 할당 하려고 하는 것이다
하지만 myName
이 가리키는 것을 변경할 수 없고, 오직 myName
의 속성을 다음과 같은 다른 것으로 변경할 수 있다:
var myName = {
firstName: "Carla"
};
function myNameIs(aName){
aName.nickName = "Carlita";
}
myNameIs(myName);
console.log(myName); // Object {firstName: "Carla", nickName: "Carlita"}