ECMAScript의 변수는 '원시 값'과 '참조 값' 두 가지 타입의 데이터를 저장할 수 있다. 원시 값은 단순한 데이터이며 Undefined, Null, Boolean, 숫자, 문자열이고, 참조 값은 여러 값으로 구성된 객체로 메모리에 저장된 객체를 가리킨다.
let person = {
name: "정다혜",
age: 26
};
let copyPerson = person;
copyPerson.age = 30;
console.log(person.age); // 30
console.log(copyPerson.age); // 30
참조 값을 변수에서 다른 변수로 복사하게 되면 그 값이 객체 자체가 아니라 heap에 저장된 객체를 가리키는 포인터이기때문에 한쪽을 변경하면 다른 쪽에도 반영이된다.
위 그림처럼 다른 변수는 같은 객체를 바라보고 있는 것을 알 수 있다. 즉, =
를 사용한 객체 복사는 참조만 하는 것이다. 이러한 방식은 객체의 불변성을 깨뜨리고, 버그를 유발할 수 있다. 실제로, 작업을 하면서 단순 대입으로 변수를 복사해 버그가 유발된 경우도 있었다.
shallow copy는 말 그대로 한 depth만 복사한다.
얕은 복사란 새로운 객체에 원본 객체의 프로퍼티의 값을 정확히 복사하는 것이다. 값이 객체라면 해당 객체의 주소를 복사하게되는 것이다.
아래의 예시를 보면 이해하기가 더 쉬울 것이다.
let obj = {
a: 1,
b: 2,
c: {
c1: 1,
c2: 2
}
}
let copyObj = Object.assign({}, obj);
copyObj.a = 10;
copyObj.c.c1 = 4;
console.log(obj.a); // 1
console.log(obj.c.c1); // 4
객체의 경우 객체의 참조 값을 복사하게 되므로, 복사한 변수에서 값을 변경해도 원래 변수값까지 변경되는 것을 확인할 수 있다.
- _.clone()
var person = {
name: 'faker',
age: 25
};
var copyPerson = _.clone(person);
- $.extend()
var person = {
name: 'faker',
age: 25
};
var copyPerson = $.extend({},person);
다음과 같이 underscore.js의 _.clone()
과 jQuery의 $.extend()
를 사용해서도 얕은 복사를 할 수 있다.
Deep Copy는 원본 객체를 완전히 복사하게 된다. Shallow Copy에서 주소를 복사한 것 또한 새로운 메모리 공간을 확보해 생성하게된다.
주로 구현하는 방법은 아래와 같은 방법이 있다.
- 재귀호출로 복사
function deepCopy(obj) {
if (obj === null || typeof(obj) !== "object") {
return obj;
}
let copy = {};
for(let key in obj) {
copy[key] = deepCopy(obj[key]);
}
return copy;
}
- JSON
let copyObj = JSON.parse(JSON.stringify(obj));
Stringily 메소드는 function의 경우에 undefined로 처리하므로 주의해서 사용해야한다.
- 라이브러리(jQuery)
var person = {
name: 'faker',
age: 25
};
var copyPerson = $.extend(true, {},person); // DeepCopy flag를 true로 설정