深入JS函數(shù)中默認(rèn)參數(shù)的使用詳解
默認(rèn)參數(shù):如果沒有值或傳入值為
undefined
的情況下,默認(rèn)函數(shù)參數(shù)允許定義好的形參對默認(rèn)值進(jìn)行初始化
下面我們來探討在js中函數(shù)使用默認(rèn)參數(shù)的幾個細(xì)節(jié),包括默認(rèn)位置,默認(rèn)參數(shù)對函數(shù)的length
屬性的影響,重新創(chuàng)建的默認(rèn)參數(shù),與形參的結(jié)合使用,默認(rèn)參數(shù)的作用域以及默認(rèn)參數(shù)的暫時性死區(qū)。
ES5中的默認(rèn)參數(shù)
在ES5時,還不能直接直接在參數(shù)中指定默認(rèn)值,只能通過在函數(shù)域內(nèi)聲明變量并賦值,達(dá)到一個默認(rèn)值的效果,但是這種不會直觀得表明默認(rèn)值參數(shù)。
function fn(x, y) { var y = y || 1; return x + y; } console.log(fn(2)); // 3
就如上代碼,我們通過||
使y
為函數(shù)的默認(rèn)參數(shù),這就是ES5的實現(xiàn)方式,但是如果我們的參數(shù)是布爾值的情況,就很容易造成問題。
下面讓我們看看在ES6針對這個問題而引入的特性——全新默認(rèn)參數(shù)的誕生
全新默認(rèn)參數(shù)的誕生
ES6函數(shù)參數(shù)默認(rèn)值是可以直接在參數(shù)定義的時候進(jìn)行默認(rèn)賦值 , 它的應(yīng)用優(yōu)化了代碼結(jié)構(gòu),也增強(qiáng)了代碼閱讀者的體驗。
function fn(x, y=1) { return x + y; } console.log(fn(2)); // 3
值得注意的是參數(shù)定義的時候賦默認(rèn)值需要注意一些細(xì)節(jié)。
默認(rèn)位置
默認(rèn)參數(shù)值的位置一定為尾參數(shù),即參數(shù)定義的尾部,下面的代碼說明了如果默認(rèn)參數(shù)的定義不放在尾部的情況
function fn(x = 1, y) { return [x, y]; } console.log(fn(2)); // [ 2, undefined ]
這種情況即我們無法再通過控制x
沒有值使用x
的默認(rèn)值,但是可以使用undefined
來解決這個問題
function fn(x = 1, y) { return [x, y]; } console.log(fn(undefined,2)); // [ 1, 2 ]
即使這樣還是建議將默認(rèn)參數(shù)定義在參數(shù)的尾部,其中一個原因就是它會影響函數(shù)length
屬性
默認(rèn)參數(shù)對函數(shù)的length
屬性的影響
函數(shù)的length
屬性會返回函數(shù)的參數(shù)數(shù)量,其不包括默認(rèn)參數(shù),但是這里有一個問題,它僅僅只會返回參數(shù)區(qū)域內(nèi)默認(rèn)參數(shù)前的形式參數(shù)數(shù)量
console.log(function fn(x, y = 1, z) {}.length); // 1
上面的函數(shù)x
被設(shè)默認(rèn)值,所以其length
屬性只返回x
之前的形式參數(shù)數(shù)量,由此還是建議將默認(rèn)參數(shù)放在參數(shù)定義的尾部
重新創(chuàng)建的默認(rèn)參數(shù)
函數(shù)在被調(diào)用的時候,參數(shù)會被創(chuàng)建,而默認(rèn)參數(shù)有一個問題,就是他是否會因為函數(shù)的重復(fù)調(diào)用而被重新創(chuàng)建,在JS
中是會的,來看下面的例子
function fn(x, num = []) { num.push(x); return num; } console.log(fn(1)); // [1] console.log(fn(2)); // [2]
通過上面兩次調(diào)用fn
函數(shù)并為num
添加元素,而每次調(diào)用得到的結(jié)果證明了每次調(diào)用函數(shù)都會重新創(chuàng)建默認(rèn)num
。
默認(rèn)參數(shù)與形參的結(jié)合使用
我們可以將默認(rèn)參數(shù)與形參結(jié)合起來使用,如下
function fn(x, y, z = [x, y, 1]) { return z; } console.log(fn(1, 2)); // [ 1, 2, 1 ]
我們通過尾部的默認(rèn)參數(shù)與前面的形式參數(shù)結(jié)合,可以是字符串形式,也可以是數(shù)組形式,也可以表達(dá)式,比如:fn(x, y, z = x + y)
默認(rèn)參數(shù)的作用域問題
深入默認(rèn)參數(shù)的原理,其本質(zhì)為使用let
聲明默認(rèn)參數(shù),所以這就造成了一個現(xiàn)象,即當(dāng)我們使用默認(rèn)參數(shù)的時候,會開辟一個單獨(dú)的作用域,在作用域內(nèi)訪問一個變量的時候會進(jìn)行一些步驟,為了更好的理解參數(shù)作用域,拿下面拿一段代碼說明這個過程:
let x = 1 function() fn(y = x) { return y } console.log(fn()) // 1
由于y = x
是默認(rèn)參數(shù),所以參數(shù)定義的區(qū)域形成了一個單獨(dú)作用域,在賦值的過程中,它將與引擎進(jìn)行如下對話,這里忽略掉LHS與RHS查詢,不是本文的要點
- 引擎在為
y
賦值的時候會詢問當(dāng)前作用域是否見過x
,當(dāng)前作用域即參數(shù)聲明的作用域回答沒有 - 引擎會向上走,詢問上級作用域,此時為全局作用域,回答有
x
,引擎為y
賦值
通過這個簡單的例子可以理解默認(rèn)參數(shù)的作用域是什么,其實它很常見,比如說for
循環(huán)
for(let i = 0; i < 100; i++) {}
默認(rèn)參數(shù)的隱藏性死區(qū)(隱藏的暫時性死區(qū))
在之前let與const命令這篇文章中提到了暫時性性死區(qū)的概念,而因為默認(rèn)參數(shù)的特性,也會造成暫時性死區(qū)的現(xiàn)象
暫時性死區(qū):在代碼塊中使用let
聲明變量前,該變量不可用,這種現(xiàn)象稱為暫時性死區(qū)
var x = 1 function fn(x = x) { return x; }
在參數(shù)內(nèi)x = x
其本質(zhì)為let x = x
在= x
的時候會出現(xiàn)暫時性死區(qū)的現(xiàn)象,因為在此刻出現(xiàn)問題的是=
后的x
,此時對x
進(jìn)行訪問,所以,而此刻在該作用域內(nèi)let x
之前,沒有聲明x
,所以該變量不可用,即我們所說的暫時性死區(qū)的現(xiàn)象。
參考文獻(xiàn)
- ES6標(biāo)準(zhǔn)入門
- 你不知道的javascript上卷
- MDN
到此這篇關(guān)于深入JS函數(shù)中默認(rèn)參數(shù)的使用詳解的文章就介紹到這了,更多相關(guān)JS函數(shù)默認(rèn)參數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
微信小程序中使用自定義字體的實現(xiàn)與體驗優(yōu)化
由于微信支持的字體非常有限,不能滿足個性化的需求,因此在開發(fā)的過程中可能會需要使用自定義字體,下面這篇文章主要給大家介紹了關(guān)于微信小程序中使用自定義字體的實現(xiàn)與體驗優(yōu)化的相關(guān)資料,需要的朋友可以參考下2022-02-02