JavaScript?新提案optional?chaining可選鏈屬性原理詳解
Optional chaining
這是一項(xiàng)新的提案,老舊瀏覽器可能需要 polyfills。
Optional chaining ?.
(可選鏈,以下簡(jiǎn)稱 OC)是一種以安全的方式去訪問(wèn)嵌套的對(duì)象屬性,即使某個(gè)屬性根本就不存在。
起因:
一個(gè)在 JavaScript 中很常見(jiàn)的問(wèn)題就是:假如用戶信息中,地址是可填可不填的,那我們就無(wú)法安全地訪問(wèn)地址的某一個(gè)屬性:
let user = {}; // 用戶可能沒(méi)有填地址 alert(user.address.street); // 報(bào)錯(cuò)
或者下面這種情況,在 WEB 開(kāi)發(fā)中,我們可能需要去獲取一個(gè) DOM 元素,但是這個(gè) DOM 元素可能不存在:
// 當(dāng) querySelector(...) 的結(jié)果為 null 的時(shí)候,程序會(huì)報(bào)錯(cuò) let html = document.querySelector('.my-element').innerHTML;
在 OC ?.
出現(xiàn)之前,我們一般通過(guò)邏輯與操作來(lái)解決:
let user = {}; // 沒(méi)有地址的用戶 alert( user && user.address && user.address.street ); // undefined (不會(huì)報(bào)錯(cuò))
使用邏輯與操作符可以確保表達(dá)式的所有部分都能夠正確執(zhí)行,但是寫(xiě)法卻比較笨重。
Optional chaining(可選鏈)
OC ?.
能夠使代碼變得簡(jiǎn)便,當(dāng)位于其前的值為 undefined
或者是 null
時(shí),會(huì)立即阻止代碼的執(zhí)行,并且返回 undefined
。
通過(guò) OC,我們可以安全地訪問(wèn)用戶的地址:
let user = {}; // 一個(gè)沒(méi)有地址的用戶 alert( user?.address?.street ); // undefined (不會(huì)報(bào)錯(cuò))
即使 user
對(duì)象不存在,使用 OC 訪問(wèn)它的地址屬性也不會(huì)報(bào)錯(cuò):
let user = null; alert( user?.address ); // undefined alert( user?.address.street ); // undefined
需要注意的是,OC 僅允許它前面相鄰的部分為可選項(xiàng),不包括進(jìn)一步的值。
在上面的代碼中,user?.
僅允許 user
這個(gè)對(duì)象為 null
或 undefined
。
另一方面,如果 user
這個(gè)對(duì)象真的存在,那么 user.address
這個(gè)屬性必須存在,否則訪問(wèn) user?.address.street
則會(huì)在第二個(gè)點(diǎn)這報(bào)錯(cuò)。
不要過(guò)度使用 OC
我們應(yīng)該僅在希望某個(gè)值可能不存在的情況下才使用 ?.
例如,根據(jù)我們的代碼邏輯,user
對(duì)象必須存在,但是 address
屬性是可選的,所以 user.address?.street
才是更好的選擇。
所以,由于其他原因?qū)е碌?nbsp;user
對(duì)象為 undefined
的情況才能被快速發(fā)現(xiàn)。否則,bug 將會(huì)變得比較難找
位于 ?.
前的變量必須被顯示聲明
如果 user
這個(gè)變量根本沒(méi)有被聲明,那么 user?.anything
將會(huì)觸發(fā)一個(gè)錯(cuò)誤:
// ReferenceError: user is not defined user?.address;
此處必須有變量聲明語(yǔ)句 let/const/var
, OC 對(duì)未聲明的變量無(wú)效
短路
在上文提到,在OC ?.
的前部分的值為 null
或 undefined
的時(shí)候,會(huì)立即停止執(zhí)行。
所以,如果在其后面如果有函數(shù)的調(diào)用,或者其他操作,都不會(huì)執(zhí)行。
let user = null; let x = 0; user?.sayHi(x++); // 什么都不會(huì)做 alert(x); // 0,值沒(méi)有自增
其他用法:?.(),?.[]
OC 不是一個(gè)操作符,而是一個(gè)特別的語(yǔ)法糖,所以可以和函數(shù)調(diào)用和中括號(hào)共用。
例如,?.()
可以用于執(zhí)行一個(gè)可能不存在的函數(shù)。
下面的代碼中,一些用戶擁有 admin
方法,一些用戶沒(méi)有:
let user1 = { admin() { alert("I am admin"); } }; let user2 = {}; user1.admin?.(); // I am admin user2.admin?.();
我們首先用點(diǎn)語(yǔ)法去獲取 admin
方法,因?yàn)?user 對(duì)象肯定存在,所以我們可以安全地去訪問(wèn);
然后使用 ?.()
檢查其左側(cè),如果 admin 這個(gè)方法存在,就執(zhí)行(例如 user1),否則就停止執(zhí)行,但是不報(bào)錯(cuò)(例如 user2)。
?.
語(yǔ)法同樣可以在當(dāng)我們需要通過(guò)中括號(hào)去訪問(wèn)屬性時(shí)使用,使用它可以安全的訪問(wèn)一個(gè)或許還不存在的對(duì)象的屬性:
let user1 = { firstName: "John" }; let user2 = null; // 假如現(xiàn)在不能授權(quán)這個(gè)用戶 let key = "firstName"; alert( user1?.[key] ); // John alert( user2?.[key] ); // undefined alert( user1?.[key]?.something?.not?.existing ) // undefined
?.
同樣可以與 delete
操作符共用
delete user?.name; // 刪除用戶名,如果用戶存在的話
使用 ?.
可以進(jìn)行刪除和讀取操作,但是不能進(jìn)行賦值操作
在賦值運(yùn)算符的左側(cè),?.
無(wú)效
// 下面的代碼是假設(shè)現(xiàn)在需要在用戶存在的情況下,重新賦值 name 屬性 user?.name = "John"; // 報(bào)錯(cuò) // 等同于 undefined = "John"
總結(jié)
OC ?.
有三種形式:
obj?.prop
- 如果obj
存在的話,返回obj.prop
的值,否則返回undefined
;obj?.[prop]
- 如果obj
存在的話返回obj[prop]
的值,否則返回undefined
;obj?.method()
- 如果obj
存在的話則調(diào)用obj.method()
方法,否則返回undefined
。
這幾種形式都非常的直觀并且易于使用。?.
檢查左側(cè)的值是否為 null
或 undefine
,如果不是的話則繼續(xù)執(zhí)行。
通過(guò)鏈?zhǔn)降?nbsp;?.
可以安全地訪問(wèn)嵌套的對(duì)象。
我們應(yīng)該僅在當(dāng)左側(cè)的值可能不存在的情況下才使用 ?.
,這樣在發(fā)生錯(cuò)誤的時(shí)候才能更加容易找到問(wèn)題。
參考鏈接:
https://javascript.info/optio...
https://developer.mozilla.org...
https://tc39.es/ecma262/#prod...
以上就是JavaScript 新提案optional chaining可選鏈的詳細(xì)內(nèi)容,更多關(guān)于JavaScript可選鏈的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
如何動(dòng)態(tài)加載外部Javascript文件
這篇文章主要介紹了如何動(dòng)態(tài)加載外部Javascript文件,本文舉例講解使用js加載器動(dòng)態(tài)加載外部Javascript文件,感興趣的小伙伴們可以參考一下2015-12-12javascript動(dòng)態(tài)創(chuàng)建對(duì)象的屬性詳解
在本篇文章中我們給大家分享了關(guān)于javascript動(dòng)態(tài)創(chuàng)建對(duì)象的屬性的相關(guān)知識(shí)點(diǎn)以及代碼分享,需要的朋友們參考學(xué)習(xí)下。2018-11-11JavaScript實(shí)現(xiàn)找出數(shù)組中最長(zhǎng)的連續(xù)數(shù)字序列
這篇文章主要介紹了JavaScript實(shí)現(xiàn)找出數(shù)組中最長(zhǎng)的連續(xù)數(shù)字序列的方法,需要的朋友可以參考下2014-09-09JS簡(jiǎn)單獲取并修改input文本框內(nèi)容的方法示例
這篇文章主要介紹了JS簡(jiǎn)單獲取并修改input文本框內(nèi)容的方法,結(jié)合實(shí)例形式分析了JavaScript針對(duì)頁(yè)面元素的獲取、賦值等相關(guān)操作技巧,需要的朋友可以參考下2018-04-04layui實(shí)現(xiàn)數(shù)據(jù)表格table分頁(yè)功能(ajax異步)
這篇文章主要為大家詳細(xì)介紹了layui實(shí)現(xiàn)數(shù)據(jù)表格table分頁(yè)功能、異步加載,表格渲染,含條件查詢,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-07-07javascript驗(yàn)證只能輸入數(shù)字和一個(gè)小數(shù)點(diǎn)示例
使用javascript限制只能輸入數(shù)字和一個(gè)小數(shù)點(diǎn),在某些情況下還是比較使用的,下面有個(gè)不錯(cuò)的示例,感興趣的朋友可以參考下2013-10-10