[ES6] Spread and Rest
Spread
Spread
는 변수를 가져와서 풀어헤치고 전개해주는 것을 말합니다.
unpacking
이라고 하죠.
Spread
를 사용하기 위해서 ...
을 사용합니다.
const fruits = ["apple", "banana", "melon", "strawberry"];
console.log(fruits);
console.log(...fruits);
// [ 'apple', 'banana', 'melon', 'strawberry' ]
// apple banana melon strawberry
이는 두 개의 array 를 합칠 때도 유용하게 사용합니다.
Spread
를 사용하지 않아도 배열을 합치는 방법은 많겠지만 코드 길이나 가독성 면에서 Spread
가 상당히 우수합니다.
const fruits = ["apple", "banana", "melon", "strawberry"];
const vegtables = ["carrot", "cucumber"]
console.log([...fruits, ...vegtables]);
// [ 'apple', 'banana', 'melon', 'strawberry', 'carrot', 'cucumber' ]
중간에 다른 값을 추가하더라도 문제 없습니다.
const fruits = ["apple", "banana", "melon", "strawberry"];
const vegtables = ["carrot", "cucumber"]
console.log([...fruits, "onion", ...vegtables]);
// [ 'apple', 'banana', 'melon', 'strawberry', 'onion', 'carrot', 'cucumber' ]
array 뿐 아니라 object 에서도 Spread
는 유효합니다.
const settings = {
theme: "light"
}
const mySettings = {
font_size: 24
}
console.log({...settings, ...mySettings});
// { theme: 'light', font_size: 24 }
Conditional Spread
다음과 같은 상황을 가정해보겠습니다.
fontSize
가 128 이하인 경우에는 이를 settings
에서 지정해주고 그렇지 않은 경우에는 지정을 하지 않는다고 해봅시다.
const settings = {
theme: "light",
fontSize: fontSize <= 128 ? fontSize : null
}
console.log(settings)
// fontSize = 30 인 경우
// { theme: 'light', fontSize: 30 }
// fontSize = 130 인 경우
// { theme: 'light', fontSize: null }
다음과 같이 삼항연산자를 통해서 구현했지만 key 값으로 fontSize
가 남아있습니다.
이 key 값 조차 없애고 싶으면 어떻게 해야할까요?
이를 optional object property
라고 합니다.
이 경우 Spread
가 유용하게 사용 될 수 있습니다.
우선 &&
을 기준으로 앞 쪽은 원하는 조건입니다.
뒤 쪽은 조금 복잡하죠. 우선 ...
으로 Spread
하기 위해서 { }
로 감싸주었습니다.
중괄호 안에는 fontSize: fontSize
처럼 써야 할 것 같지만 한 번만 적어줘도 문제 없습니다.
shorthand
덕분이죠. shorthand가 뭔지 모른다면 아래 링크를 참조하시길 바랍니다.
https://ssungkang.tistory.com/entry/ES6-Destructuring
const settings = {
theme: "light",
...(fontSize <= 128 && {fontSize})
}
console.log(settings)
// fontSize = 30 인 경우
// { theme: 'light', fontSize: 30 }
// fontSize = 130 인 경우
// { theme: 'light' }
Rest Parameters
위에서 살펴봤던 Spread
가 unpacking
을 해주었다면 Rest
는 반대로 packing
을 해줍니다.
자기소개를 하는 함수를 만들어봅시다. 이 경우 함수의 파라미터가 너무 많기 때문에 이를 다 기재해주는 것은 효율적이지 못합니다.
이런 경우 Rest
를 사용하며 한 번에 받을 수 있습니다.
const intro = (...info) => console.log(info);
intro("minsung", 25, "Seoul", "UOS", ["basketball", "go"]);
// [ 'minsung', 25, 'Seoul', 'UOS', [ 'basketball', 'go' ] ]
물론 필요한 부분은 따로 받고 나머지에 한하여 묶어서 받아줄 수도 있습니다.
const intro = (name,...rest) => {
console.log(`my name is ${name}`);
console.log(rest);
}
intro("minsung", 25, "Seoul", "UOS", ["basketball", "go"]);
// my name is minsung
// [ 25, 'Seoul', 'UOS', [ 'basketball', 'go' ] ]
Rest
로 값을 받으면 출력값의 공통점이 있습니다.
결과값이 array로 출력 된다는 것이죠.
Rest + Spread + Destructure
지금까지 배운 세 가지를 활용하면 많은 것을 할 수 있습니다.
한 번 알아보도록 하죠.
Destructure
가 익숙하지 않다면 아래를 참고해주세요.
https://ssungkang.tistory.com/entry/ES6-Destructuring
delete property
우선 첫 번째 예시는 user object 에 대해 익명으로 바꿔주고 싶습니다.
Destructure
를 이용해서 값을 받는데 이 때 Rest
를 통해 name 값과 나머지 값으로 받아준 후 나머지 값에 대해서만 반환해주었습니다.
이를 잘 활용하여 object 에 필요없는 key 들을 지울 수 있습니다.
const user = {
name: "minsung",
age: 25,
city: "Seoul",
school: "UOS"
}
const deleteName = ({name, ...rest}) => rest;
const AnonymousUser = deleteName(user);
console.log(AnonymousUser);
// { age: 25, city: 'Seoul', school: 'UOS' }
insert property
이번엔 필요한 key 를 넣어봅시다.
일반적으로 object 에 바로 넣게 되면 기존의 있는 값이 overwrite 될 수 있기 때문에 default value 를 설정하기가 까다롭습니다.
이 경우에도 Destructure
와 Rest
를 사용하면 해결 가능합니다.
7번째 줄의 ...rest
가 두 번 등장하는데 서로 다릅니다. 함수의 파리미터에 위치하는 첫 번째는 Rest
고 반환해주는 곳에 위치하는 두 번째는 Spread
에 해당됩니다.
let user = {
name: "minsung",
age: 25,
city: "Seoul",
}
const insertSchool = ({school="UOS", ...rest}) => ({school, ...rest});
user = insertSchool(user);
console.log(user);
// { school: 'UOS', name: 'minsung', age: 25, city: 'Seoul' }
rename property
이번에는 key 값을 바꿔보도록 하겠습니다.
위와 동일한 논리로 전개됩니다.
다만 추가된 부분은 Destructure
의 rename 이 사용되었습니다.
let user = {
name: "minsung",
age: 25,
city: "Seoul",
}
const rename = ({name:nickname, ...rest}) => ({nickname, ...rest});
user = rename(user);
console.log(user);
// { nickname: 'minsung', age: 25, city: 'Seoul' }