在ES5與ES6環(huán)境下處理函數(shù)默認(rèn)參數(shù)的實(shí)現(xiàn)方法
函數(shù)默認(rèn)值是一個(gè)很提高魯棒性的東西(就是讓程序更健壯)
MDN關(guān)于函數(shù)默認(rèn)參數(shù)的描述:函數(shù)默認(rèn)參數(shù)允許在沒有值或undefined被傳入時(shí)使用默認(rèn)形參。
ES5
使用邏輯或||來(lái)實(shí)現(xiàn)
眾所周知,在ES5版本中,并沒有提供的直接方法供我們我們處理函數(shù)默認(rèn)值
所以只能夠自己去增強(qiáng)函數(shù)的功能,一般會(huì)這么來(lái)做:
function doSomething (name, age) { name = name || 'default name' age = age || 18 console.log(name, age) }
我們將函數(shù)的兩個(gè)參數(shù)name與age進(jìn)行默認(rèn)值的處理,如果沒有則使用默認(rèn)值。
在執(zhí)行一下函數(shù)后,好像并沒有什么不對(duì):
doSomething() // default name, 18 doSomething('Niko') // Niko , 18 doSomething(, 12) // default name, 12
然而當(dāng)我們執(zhí)行這樣的代碼時(shí),就會(huì)獲得一些超出預(yù)期的結(jié)果:
doSomething('Niko', 0) // Niko, 18
能夠發(fā)現(xiàn),對(duì)于參數(shù)0,我們上邊的默認(rèn)參數(shù)實(shí)現(xiàn)方法是有問題的
就像下邊的四個(gè)表達(dá)式,都會(huì)輸出wrong,這很顯然不能夠滿足上邊MDN關(guān)于函數(shù)默認(rèn)參數(shù)的定義:
console.log(0 || 'wrong') console.log('' || 'wrong') console.log(null || 'wrong') console.log(false || 'wrong')
正確的姿勢(shì)
所以,在ES5中正確的默認(rèn)值處理應(yīng)該是這樣:
function doSomething (name, age) { if (name === undefined) { name = 'default name' } if (age === undefined) { age = 18 } console.log(name, age) }
使用三元運(yùn)算符簡(jiǎn)化操作
或者我們簡(jiǎn)寫成三元運(yùn)算符形式的:
function doSomething (name, age) { name = name === undefined ? 'default name' : name age = age === undefined ? 18 : age console.log(name, age) }
使用函數(shù)進(jìn)行封裝
但是如果我們每寫一個(gè)函數(shù),都要重復(fù)的去做這些操作
未免太麻煩了,所以,我們對(duì)這個(gè)邏輯進(jìn)行一個(gè)簡(jiǎn)單的封裝:
function defaultValue (val, defaultVal) { return val === undefined ? defaultVal : val } function doSomething (name, age) { name = defaultValue(name, 'default name') age = defaultValue(age , 18) console.log(name, age) }
這樣就很簡(jiǎn)潔的在ES5實(shí)現(xiàn)了函數(shù)默認(rèn)參數(shù)的邏輯
one momre things
關(guān)于上邊的defaultValue函數(shù)實(shí)現(xiàn)方法,我們?cè)诤侠淼氖褂萌躅愋驼Z(yǔ)言的優(yōu)勢(shì)后
可以使用這種方式來(lái)省去三元運(yùn)算符的操作:
function defaultValue () { return arguments[+(arguments[0] === undefined)] }
我們知道,arguments表示函數(shù)所有的實(shí)參
我們使用arguments[0]獲取第一個(gè)實(shí)參,然后與undefined進(jìn)行全等比較
在外層將表達(dá)式的結(jié)果轉(zhuǎn)換為Number,然后將這個(gè)值作為下標(biāo)獲取arguments中對(duì)應(yīng)的參數(shù)。
因?yàn)槭怯葿oolean值轉(zhuǎn)變而來(lái),所以只會(huì)存在0、1兩種選項(xiàng)。
也就實(shí)現(xiàn)了上邊三元運(yùn)算符的功能。
ES6
ES6版本的函數(shù)默認(rèn)值基本上就是我們上邊實(shí)現(xiàn)的那種套路了
但是因?yàn)槭窃?,所以?huì)有相應(yīng)的新語(yǔ)法,能夠更簡(jiǎn)潔的使用:
function doSomething (name = 'default name', age = 18) { console.log(name, age) }
ES6中提供了新的語(yǔ)法,可以讓我們?cè)诤瘮?shù)聲明參數(shù)后邊直接寫= [defaultValue]的這種形式來(lái)設(shè)置某個(gè)參數(shù)的默認(rèn)值。
直接使用這種方式,省去了在函數(shù)內(nèi)部進(jìn)行默認(rèn)值的檢查,能夠讓函數(shù)專注的做它應(yīng)該做的事情。
如何針對(duì)某些必填參數(shù)拋出異常
ES6這種新語(yǔ)法能夠讓我們很好的針對(duì)某個(gè)必填參數(shù)進(jìn)行錯(cuò)誤提醒:
function requireParams () { throw new Error('required params') } function doSomething (name = requireParams(), age = 18) { // do something }
如果name參數(shù)為undefined,就會(huì)觸發(fā)默認(rèn)值規(guī)則
然后調(diào)用requireParams函數(shù),而我們?cè)诤瘮?shù)中直接throw了一個(gè)Error
復(fù)雜結(jié)構(gòu)參數(shù)的默認(rèn)值處理
上邊的處理都是針對(duì)簡(jiǎn)單的基本類型數(shù)據(jù)進(jìn)行處理的,但如果我們有如下的一個(gè)函數(shù):
function init ({id, value}) {} init({ id: 'tagId', value: 1 })
如果在ES5環(huán)境下,針對(duì)這種參數(shù)的默認(rèn)值處理將會(huì)變得無(wú)比復(fù)雜
首先要判斷這一個(gè)參數(shù)是否存在,然后在判斷參數(shù)中的所有key是否存在
而在ES6中,可以這樣來(lái)做:
function init ({ id = 'defaultId', value = 1 } = {}) { console.log(id, value) } init()
首先在解構(gòu)函數(shù)的后邊添加默認(rèn)值= {},然后針對(duì)每一項(xiàng)參數(shù)添加默認(rèn)值,很簡(jiǎn)潔的就實(shí)現(xiàn)了我們的需求。
ES5版本的polyfill代碼在倉(cāng)庫(kù)中的位置:defaultValue
參考資料
1.MDN
總結(jié)
以上所述是小編給大家介紹的在ES5與ES6環(huán)境下處理函數(shù)默認(rèn)參數(shù)的實(shí)現(xiàn)方法,希望對(duì)大家有所幫助,如果大家有任何疑問歡迎給我留言,小編會(huì)及時(shí)回復(fù)大家的!
- es6中的解構(gòu)賦值、擴(kuò)展運(yùn)算符和rest參數(shù)使用詳解
- ES6中的rest參數(shù)與擴(kuò)展運(yùn)算符詳解
- ES6知識(shí)點(diǎn)整理之函數(shù)對(duì)象參數(shù)默認(rèn)值及其解構(gòu)應(yīng)用示例
- ES6知識(shí)點(diǎn)整理之函數(shù)數(shù)組參數(shù)的默認(rèn)值及其解構(gòu)應(yīng)用示例
- 深入淺出ES6新特性之函數(shù)默認(rèn)參數(shù)和箭頭函數(shù)
- ES6中箭頭函數(shù)的定義與調(diào)用方式詳解
- ES6中的箭頭函數(shù)實(shí)例詳解
- 關(guān)于ES6箭頭函數(shù)中的this問題
- ES6記錄異步函數(shù)的執(zhí)行時(shí)間詳解
- Es6 Generator函數(shù)詳細(xì)解析
- ES6新特性之函數(shù)的擴(kuò)展實(shí)例詳解
- es6函數(shù)之rest參數(shù)用法實(shí)例分析
相關(guān)文章
js實(shí)現(xiàn)類似新浪微博首頁(yè)內(nèi)容漸顯效果的方法
這篇文章主要介紹了js實(shí)現(xiàn)類似新浪微博首頁(yè)內(nèi)容漸顯效果的方法,實(shí)例分析了漸顯效果的實(shí)現(xiàn)要點(diǎn)與方法,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-04-04小議Function.apply() 之一------(函數(shù)的劫持與對(duì)象的復(fù)制)
小議Function.apply() 之一------(函數(shù)的劫持與對(duì)象的復(fù)制)...2006-11-11js實(shí)現(xiàn)飛機(jī)大戰(zhàn)游戲
這篇文章主要為大家詳細(xì)介紹了js實(shí)現(xiàn)飛機(jī)大戰(zhàn)游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-08-08RequireJS多頁(yè)面應(yīng)用實(shí)例分析
這篇文章主要介紹了RequireJS多頁(yè)面應(yīng)用實(shí)例分析的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-06-06JavaScript實(shí)現(xiàn)留言板實(shí)戰(zhàn)案例
這篇文章主要給大家介紹了關(guān)于JavaScript實(shí)現(xiàn)留言板的相關(guān)資料,使用JavaScript來(lái)編寫留言板功能相信大家都不陌生,文中給出了詳細(xì)的示例代碼,需要的朋友可以參考下2023-07-07一步一步的了解webpack4的splitChunk插件(小結(jié))
這篇文章主要介紹了一步一步的了解webpack4的splitChunk插件(小結(jié)),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來(lái)看看吧2018-09-09原生JS實(shí)現(xiàn)列表內(nèi)容自動(dòng)向上滾動(dòng)效果
這篇文章主要介紹了原生JS實(shí)現(xiàn)列表內(nèi)容自動(dòng)向上滾動(dòng)效果,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-05-05