關(guān)于JS解構(gòu)的5種有趣用法
前言
原文標(biāo)題:5 Interesting Uses of JavaScript Destructuring
原文鏈接:dmitripavlutin.com/5-interesti…
定期回顧我寫(xiě)的JS代碼,我發(fā)現(xiàn)解構(gòu)運(yùn)算無(wú)處不在。
獲取對(duì)象的屬性和訪問(wèn)數(shù)組內(nèi)容是都是很常用的操作。而解構(gòu)運(yùn)算使得這些操作變得非常簡(jiǎn)單明了。
在這篇文章中,我將會(huì)講解JS解構(gòu)不同于常見(jiàn)用法的五種使用技巧。
1. 交換變量
常見(jiàn)的交互兩個(gè)變量值的方法都需要借助一個(gè)額外的變量,看一個(gè)簡(jiǎn)單的例子:
let a = 1; let b = 2; let temp; temp = a; a = b; b = temp; a; // => 2 b; // => 1
temp是一個(gè)臨時(shí)變量,在例子中存儲(chǔ)了變量a的值,b的值賦給了a,最后把temp的值賦給了b。
解構(gòu)運(yùn)算使得交換變量的值變得非常簡(jiǎn)單,不需要借助第三個(gè)臨時(shí)變量:
let a = 1; let b = 2; [a, b] = [b, a]; a; // => 2 b; // => 1
[a, b] = [b, a]是一個(gè)解構(gòu)運(yùn)算。在等號(hào)的右側(cè),創(chuàng)建了一個(gè)數(shù)組[b, a],對(duì)應(yīng)的值為[2, 1]。數(shù)組的第一個(gè)值2被解構(gòu)賦值給了a,第二項(xiàng)1被解構(gòu)賦值給了b。
即使這種方式仍然創(chuàng)建了一個(gè)臨時(shí)數(shù)組,但是解構(gòu)賦值對(duì)于交換變量的值仍然是非常高效簡(jiǎn)單的方式。
這種方式并沒(méi)有什么限制。你還可以同時(shí)交互更多的變量值,比如:
let zero = 2; let one = 1; let two = 0; [zero, one, two] = [two, one, zero]; zero; //=> 0 one; //=> 1 two; //=> 2
你可以交換任意數(shù)量的變量值,只是兩個(gè)變量值的交換的情況更加常見(jiàn)。
2. 訪問(wèn)數(shù)組
有一個(gè)數(shù)組,這個(gè)數(shù)組有可能是空的。有一種需求是訪問(wèn)任意位置數(shù)組元素,如果這個(gè)位置為空,則返回一個(gè)默認(rèn)值。
通常情況下有的人可能會(huì)使用數(shù)組的length屬性做判斷:
const colors = [];
let firstColor = "white";
if (colors.length > 0) {
firstColor = colors[0];
}
firstColor; //=> "white"
幸運(yùn)的是,數(shù)組解構(gòu)可以更快捷高效的實(shí)現(xiàn)相同的效果:
const colors = []; const [firstColor = "white"] = colors; firstColor; //=> "white"
const [firstColor = "white"] = colors;將colors數(shù)組的第一個(gè)元素賦值給了變量firstColor。如果這個(gè)數(shù)組的下標(biāo)為0的位置上沒(méi)有任何元素(注:為undefined時(shí)即認(rèn)為為空),white將作為默認(rèn)值賦值給firstColor。
數(shù)組解構(gòu)是非常靈活的,如果你只想訪問(wèn)數(shù)組的第二個(gè)元素,方法如下所示:
const colors = []; const [, secondColor = "black"] = colors; secondColor; //=> "black"
在解構(gòu)表達(dá)式的左邊寫(xiě)一個(gè)逗號(hào):意味著數(shù)組的第一個(gè)元素被忽略掉。colors數(shù)組下標(biāo)為1的元素被解構(gòu)賦值給了變量secondColor。
3. 不可變操作
從我開(kāi)始使用React,到后來(lái)的Redux,我被迫開(kāi)始寫(xiě)一些遵循不可變?cè)瓌t的代碼。剛開(kāi)始的時(shí)候確實(shí)有點(diǎn)不適應(yīng),不過(guò)后來(lái)我就意識(shí)到了這種方式的好處:它使得處理單向數(shù)據(jù)流更加容易。
不可變?cè)瓌t禁止修改對(duì)象。幸運(yùn)的是,解構(gòu)可以幫助你在遵循不可變?cè)瓌t的同時(shí)完成這些操作。
將解構(gòu)與展開(kāi)運(yùn)算符(rest operator)結(jié)合使用來(lái)移除數(shù)組的第一個(gè)元素:
const numbers = [1,2,3]; const [, ...fooNumbers] = numbers; fooNumbers; //=> [2, 3] numbers; //=> [1,2,3]
這個(gè)解構(gòu)操作[, ...fooNumbers] = numbers創(chuàng)建了一個(gè)新的數(shù)組fooNumbers,這個(gè)數(shù)組包含numbers除了第一個(gè)元素外的其余元素。
numbers數(shù)組并沒(méi)有被改變,這種方式遵循了不可變?cè)瓌t。
除此之外,你也可以在遵循不可變?cè)瓌t的同時(shí)使用同樣的方法來(lái)刪除一個(gè)對(duì)象的屬性。如下所示,刪除big對(duì)象的foo屬性:
const big = {
foo: "value foo",
bar: "value bar",
}
const { foo, ...small } = big;
small; //=> { bar: "value bar" }
big; //=>{ foo: "value foo", bar: "value bar" }
上述方法將解構(gòu)與對(duì)象展開(kāi)運(yùn)算符結(jié)合起來(lái)使用,創(chuàng)建了一個(gè)新的對(duì)象small,這個(gè)新對(duì)象包含big對(duì)象除了foo屬性之外的所有屬性。
4. 解構(gòu)可迭代的值
在前面幾部分內(nèi)容中,都是解構(gòu)的數(shù)組。實(shí)際上解構(gòu)運(yùn)算是可以用于所有的可迭代對(duì)象的。
許多原生的基礎(chǔ)類(lèi)型和對(duì)象都是可迭代的,例如數(shù)組,類(lèi)數(shù)組,字符串,set集合和map集合。
例如,你可以把字符串解構(gòu)成單個(gè)字符:
const str = "cheese"; const [firstChar = ""] = str; firstChar; //=> 'c'
當(dāng)然解構(gòu)不僅僅限于原生可迭代的那幾種類(lèi)型。解構(gòu)可以被用于所有實(shí)現(xiàn)了迭代接口(iterable protocol)的對(duì)象。
如下所示,movies包含一個(gè)movie對(duì)象列表。我們想要解構(gòu)movies對(duì)象的時(shí)候,可以獲取到電影的title這個(gè)字符串。實(shí)現(xiàn)這個(gè)操作首先需要自定義一個(gè)迭代器:
const movies = {
list: [
{ title: "Heat" },
{ title: "Interstellar" },
],
[Symbol.iterator]() {
let index = 0;
return {
next: () => {
if (index < this.list.length) {
const value = this.list[index++].title;
return { value, done: false };
}
return { done: true }
}
}
}
}
const [firstMovieTitle] = movies;
console.log(firstMovieTitle); //=> 'Heat'
movies對(duì)象通過(guò)定義Symbol.iterator方法實(shí)現(xiàn)了一個(gè)迭代器。這個(gè)迭代器可以迭代所有電影的title屬性。
我們?cè)趍ovies對(duì)象上遵循了迭代接口實(shí)現(xiàn),從而實(shí)現(xiàn)了通過(guò)解構(gòu)movies來(lái)獲取到標(biāo)題,比如我們獲取第一個(gè)電影的標(biāo)題:const [firstMovieTitle] = movies; 。
解構(gòu)用法的上限就是沒(méi)有上限。
5. 解構(gòu)動(dòng)態(tài)屬性
在我的經(jīng)驗(yàn)中,解構(gòu)一個(gè)對(duì)象的屬性要遠(yuǎn)比解構(gòu)一個(gè)數(shù)組的情況更多。
解構(gòu)對(duì)象看起來(lái)非常的簡(jiǎn)單:
const movie = { title: "Heat" };
const { title } = movie;
title; //=> Heat
const { title } = movie;創(chuàng)建了一個(gè)變量title,然后把movie.title的值賦值給了這個(gè)變量。
當(dāng)我第一次了解到對(duì)象解構(gòu)的時(shí)候,有一點(diǎn)令我驚訝的是你并不需要預(yù)先知道屬性的靜態(tài)名稱(chēng)。你可以通過(guò)動(dòng)態(tài)屬性名來(lái)解構(gòu)一個(gè)對(duì)象。
為了了解動(dòng)態(tài)解構(gòu)的工作原理,我們來(lái)寫(xiě)一個(gè)打招呼的函數(shù)作為例子:
function greet( obj, nameProp ) {
const { [nameProp]: name="Unknow" } = obj;
return `Hello, ${name}!`;
}
greet({ name: "Batman" }, "name"); //=> Hello, Batman!
greet( {}, "name" ); //=> Hello, Unknow!
greet()被調(diào)用時(shí)需要傳遞兩個(gè)參數(shù),一個(gè)是對(duì)象,一個(gè)是屬性名稱(chēng)。
在greet()函數(shù)內(nèi)部,解構(gòu)表達(dá)式const { [nameProp]: name="Unknow" } = obj;使用中括號(hào)[nameProp]讀取動(dòng)態(tài)屬性的名稱(chēng)。name變量接收動(dòng)態(tài)屬性的值。
更好的做法就是你可以指定一個(gè)默認(rèn)的值Unknow以防屬性不存在的情況。
6. 總結(jié)
解構(gòu)可以幫助你更方便快捷的訪問(wèn)對(duì)象屬性和數(shù)組元素。
除了基本用法之外,數(shù)組解構(gòu)還可以方便的交換變量,訪問(wèn)數(shù)組元素,做一些遵循不可變?cè)瓌t的操作。
JavaScript提供了更多的可能性,因?yàn)槟憧梢酝ㄟ^(guò)擴(kuò)展迭代器實(shí)現(xiàn)自定義的解構(gòu)邏輯。
好了,以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
JS數(shù)組操作大全對(duì)象數(shù)組根據(jù)某個(gè)相同的字段分組
這篇文章主要介紹了JS數(shù)組操作大全對(duì)象數(shù)組根據(jù)某個(gè)相同的字段分組,需要注意的是,在開(kāi)發(fā)過(guò)程這種數(shù)組的處理函數(shù),應(yīng)當(dāng)被編寫(xiě)到項(xiàng)目的公共工具函數(shù)庫(kù)中全局調(diào)用,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-11-11
javascript中"/"運(yùn)算符常見(jiàn)錯(cuò)誤
javascript里面的"/"運(yùn)算符跟C++里面的"/"運(yùn)算符不一樣,后者自動(dòng)取整,前者若非整除則會(huì)得到小數(shù)(比如說(shuō)5/2=2.5)。2010-10-10
原生js實(shí)現(xiàn)省市區(qū)三級(jí)聯(lián)動(dòng)代碼分享
這篇文章主要介紹了原生js實(shí)現(xiàn)省市區(qū)三級(jí)聯(lián)動(dòng)功能以及代碼分享,對(duì)此有需要的朋友可以參考學(xué)習(xí)下。2018-02-02
javascript日期驗(yàn)證之輸入日期大于等于當(dāng)前日期
Js使用WScript.Shell對(duì)象執(zhí)行.bat文件和cmd命令
JS中Generator函數(shù)與async函數(shù)用法介紹

