這兩個分別稱為 展開運算子(spread operator
) 及 其餘運算子 (rest operator
,也可稱為其餘參數),這兩個運算符有個兩個特點,就是都與陣列有關係,除此之外他都是 ...
;第一次看到這樣的符號出現在 JavaScript 中,我還認為這是哪個預處理器的語法,不過在此他真的是 ES6 語法,以下內容一樣可以貼到 Chrome 運行。
預處理器:像是 CSS 的 Sass,在編譯前無法直接被讀取,需要透過編譯為
.css
或.js
才能被瀏覽器使用,常見的 JavaScript 預處理器有CoffeeScript
、TypeScript
。
展開非常的好用,我們先來看一個簡單的範例,這裡有兩組的陣列,我們想把他合併到同一個變數上,那麼我們可以直接用以下方法接起來。
1 | let groupA = ['小明', '杰倫', '阿姨']; |
誒…,看到這裡是不是有似懂非懂的感覺,看似那麼容易卻不太容易理解,...
到底做了什麼事情!? 它其實一次又一次的 return 陣列中的值。
1 | console.log(...groupA); |
順帶一提,以上的概念如果用傳統的寫法會像是這樣。
1 | let groupA = ['小明', '杰倫', '阿姨']; |
另外陣列與物件相同都有著傳參考的特性,所以當把陣列賦予到另一個值上時,修改其中一個另一個也會跟著變動。
1 | // 由於傳參考的關係,所以將一個陣列傳到另一個上時 |
由於 展開運算子 它是一個一個將值寫入,所以他也有淺層的複製(shallow copy
) 。
1 | // 這個屬於淺拷貝,所以不會影響到另一個物件 |
JavaScript 中有許多類陣列,這類陣列有著陣列的外皮,但卻不能使用陣列的方法,相信先前有參考過原型章節的文章有發現這點,這類陣列由於原型不同,所以 “不能” 使用許多的陣列方法,如: map()
, concat()
等等。
其中一種很常見的就是 DOM 陣列,這也可以透過展開運算子轉為純陣列。
1 | // 可以將類陣列轉成陣列 |
在先前小明儲值的故事中,悠遊卡的儲值是使用 for...in
,原因也在於他不是真正的陣列,不過當他如果轉成真正的陣列後,就多了很多方法可以用了 (嘿嘿嘿)。1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17// 同樣道理,arguments 不是真正的陣列,也可以透過 ... 來轉成純陣列
var originCash = 1000;
function updateEasyCard() {
let arg = [...arguments]
let sum = arg.reduce(function (accumulator, currentValue) {
// 分別為前一個回傳值, 當前值
return accumulator + currentValue; // 與前一個值相加
}, 0);
// 如果使用 arguments 則會出現 `arguments.reduce is not a function`
console.log('我有 ' + sum + ' 元');
}
updateEasyCard(0); // 我有 1000 元
// arguments = [];
updateEasyCard(10, 50, 100, 50, 5, 1, 1, 1, 500); // 我有 718 元
// arguments = [10, 50, 100, 50, 5, 1, 1, 1, 500];
其餘參數,顧名思義就是傳入的參數,用途類似 arguments
,但不同的是:
arguments
不是真的陣列,其餘參數則是arguments
不能混用自訂傳入的參數以下就是使用其餘參數改寫以上範例。
1 | let mingCard = { |
透過其餘參數,我們還能混用其他的變數像是 mingCard
,讓這個儲值函式彈性更高,而不受 arguments
依序傳入的限制。
看到目前為止,你會跟我一樣覺得 ES6 的程式碼看起來不像 JavaScript 嗎?