일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
- js
- 배열
- as?
- 연산자
- ListView
- go
- golang
- 싱글 스레드
- node.js
- 노트북
- var
- Array
- Android
- 패널 교체
- 파이썬
- 안드로이드
- adapter
- javascript
- HP
- 노트북 추천
- 함수
- 자바스크립트
- 리스트 뷰
- Overloading
- Kotlin
- 오버로딩
- 자바
- Python
- 코틀린
- Java
- Today
- Total
Bbaktaeho
[JavaScript] Object.assign 그리고 Spread (Object, 객체, 속성 복사, spread) 본문
[JavaScript] Object.assign 그리고 Spread (Object, 객체, 속성 복사, spread)
Bbaktaeho 2021. 5. 11. 11:56들어가며
최근, NestJS 프레임워크로 개발하다가 공식 문서에서 assign 메서드의 예제 코드를 봤다.
Serialization 문서를 보니 생성자에 Object.assign() 메서드를 사용하고 있다.
여태 이런 꿀팁을 모르고 있어서 정리하기로 했고 스프레드 연산자를 활용한 방법과 비교해보기로 했다. (Partial은 타입스크립트 문법으로 다음에 다뤄보겠습니다)
Object.assign
타겟을 지정한 객체로 다른 객체의 속성을 복사한다.
원형
Object.assign(target, ...sources)
간단한 예제로 파악해보자. (MDN의 첫 예제를 그대로 가져왔다)
기본형 예제
const target = { a: 1, b: 2 };
const source = { b: 4, c: 5 };
const returnedTarget = Object.assign(target, source);
console.log(target);
console.log(returnedTarget);
출력은 아래와 같다.
{ a: 1, b: 4, c: 5 }
{ a: 1, b: 4, c: 5 }
기존 target 객체의 속성이 변경되거나 추가되었다.
리턴으로 받은 returnedTarget 객체는 사실 target 객체와 같은 객체를 참조하고 있다.
console.log(target === returnedTarget); // true
딱히 리턴은 받을 필요 없다.
위 예제로 알 수 있는 건 같은 속성 이름(키)이 있다면 덮어쓰기가 되며, 새로운 속성 이름이 source에 있다면 target 객체에 추가된다.
source가 여러 개 있을 때도 동일하게 동작한다.
const target = { a: 1, b: 2 };
const sources = [
{ b: 4, c: 5 },
{ b: 6, c: 7, d: 8 },
{ b: 9, c: 10, d: 11 },
{ b: 12, c: 13, d: 14, e: 15 },
];
Object.assign(target, ...sources);
console.log(target);
스프레드 연산자로 sources 리스트를 펼쳐서 인자를 넣어줘야 한다. 안 그러면 리스트의 index 값으로 키가 새로 생성되며 target에 속성이 추가된다.
// Object.assign(target, ...sources); 둘이 같다.
Object.assign(target, sources[0], sources[1], sources[2], sources[3]);
스트레드 연산자로 assign 해주면 출력은 아래와 같다.
{ a: 1, b: 12, c: 13, d: 14, e: 15 }
리스트를 그대로 assign 한다면? (...빼고 실행해보세요!)
{
'0': { b: 4, c: 5 },
'1': { b: 6, c: 7, d: 8 },
'2': { b: 9, c: 10, d: 11 },
'3': { b: 12, c: 13, d: 14, e: 15 },
a: 1,
b: 2
}
원형을 꼭 확인하고 사용하자!
참조형 예제
sources 객체들 중에 속성 값으로 참조형이 존재한다면 깊은 복사가 이루어지지 않는다. (당연하지만 주의하자)
const arr1 = [1, 2, 3];
const arr2 = [4, 5, 6];
const target = { arr: arr1 };
const source = {arr: arr2 };
Object.assign(target, source);
console.log(target.arr === source.arr); // true
깊은 복사를 원한다면 다른 방식을 활용하거나 응용하도록 하자.
Spread vs Object.assign
assign을 알기 전에 항상 스프레드 연산자를 활용하여 객체에 속성을 추가하거나 복사해줬다.
const target = { a: 1, b: 2 };
const source = { b: 4, c: 5 };
const newOb = {...target, ...source};
console.log(newOb); // { a: 1, b: 4, c: 5 }
역시나 출력은 assign과 같다.
하지만 둘은 분명한 차이점이 있었다.
불변성(immutable)을 지켜야 된다면 Object.assign을 사용하는 것 보다 스프레드 연산자를 사용하는 것이 좋다.
Object.assign으로 불변성을 지키지 못하는 것은 아닌데 target에 항상 {} 빈 객체를 인자로 전달해야 한다.
const obj = {
add: (a, b) => a + b,
};
const newObj = { ...obj };
console.log(newObj);
const copy = Object.assign({}, obj);
console.log(copy);
// obj 객체는 불변성
자세한 차이는 아래 Reference에 링크를 참조!
Reference
developer.mozilla.org/ko/docs/Web/JavaScript/Reference/Global_Objects/Object/assign
thecodebarbarian.com/object-assign-vs-object-spread.html
'프로그래밍 (Programming) > 자바스크립트 (JavaScript)' 카테고리의 다른 글
[Javascript] 기본 제공 API (자바스크립트 팁) (0) | 2020.10.24 |
---|---|
[Javascript] Generator 활용과 장점 (iterable, iterator, lazy evaluation) (0) | 2020.10.21 |
[Javascript] Generator, Iterator, Iterable (제너레이터, 이터레이터, function*) (0) | 2020.10.17 |
[Javascript] null 병합 연산자 '??' 와 OR 연산자 '||' 의 차이 (자바스크립트, nullish coalescing, falsy값) (2) | 2020.08.24 |
[Javascript/Node.js] 싱글 스레드 프로그래밍 (uncaughtExcetion) (0) | 2020.05.26 |