如何在ES5與ES6環境下處理函式預設引數

賈順名發表於2018-05-12

函式預設值是一個很提高魯棒性的東西(就是讓程式更健壯)
MDN關於函式預設引數的描述:函式預設引數允許在沒有值或undefined被傳入時使用預設形參。

ES5

使用邏輯或||來實現

眾所周知,在ES5版本中,並沒有提供的直接方法供我們我們處理函式預設值
所以只能夠自己去增強函式的功能,一般會這麼來做:

function doSomething (name, age) {
  name = name || `default name`
  age  = age  || 18

  console.log(name, age)
}

 

我們將函式的兩個引數nameage進行預設值的處理,如果沒有則使用預設值。
在執行一下函式後,好像並沒有什麼不對:

doSomething()       // default name, 18
doSomething(`Niko`) // Niko        , 18
doSomething(, 12)   // default name, 12

 

然而當我們執行這樣的程式碼時,就會獲得一些超出預期的結果:

doSomething(`Niko`, 0) // Niko, 18

能夠發現,對於引數0,我們上邊的預設引數實現方法是有問題的

就像下邊的四個表示式,都會輸出wrong,這很顯然不能夠滿足上邊MDN關於函式預設引數的定義:

console.log(0         || `wrong`)
console.log(``        || `wrong`)
console.log(null      || `wrong`)
console.log(false     || `wrong`)

 

正確的姿勢

所以,在ES5中正確的預設值處理應該是這樣:

function doSomething (name, age) {
  if (name === undefined) {
    name = `default name`
  }

  if (age === undefined) {
    age = 18
  }

  console.log(name, age)
}

 

使用三元運算子簡化操作

或者我們簡寫成三元運算子形式的:

function doSomething (name, age) {
  name = name === undefined ? `default name` : name
  age  = age  === undefined ? 18             : age

  console.log(name, age)
}

 

使用函式進行封裝

但是如果我們每寫一個函式,都要重複的去做這些操作
未免太麻煩了,所以,我們對這個邏輯進行一個簡單的封裝:

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)
}

 

這樣就很簡潔的在ES5實現了函式預設引數的邏輯

one momre things

關於上邊的defaultValue函式實現方法,我們在合理的使用弱型別語言的優勢後
可以使用這種方式來省去三元運算子的操作:

function defaultValue () {
  return arguments[+(arguments[0] === undefined)]
}

 

我們知道,arguments表示函式所有的實參
我們使用arguments[0]獲取第一個實參,然後與undefined進行全等比較
在外層將表示式的結果轉換為Number,然後將這個值作為下標獲取arguments中對應的引數。
因為是由Boolean值轉變而來,所以只會存在01兩種選項。
也就實現了上邊三元運算子的功能。

ES6

ES6版本的函式預設值基本上就是我們上邊實現的那種套路了
但是因為是原生的,所以會有相應的新語法,能夠更簡潔的使用:

function doSomething (name = `default name`, age = 18) {
  console.log(name, age)
}

 

ES6中提供了新的語法,可以讓我們在函式宣告引數後邊直接寫= [defaultValue]的這種形式來設定某個引數的預設值。
直接使用這種方式,省去了在函式內部進行預設值的檢查,能夠讓函式專注的做它應該做的事情。

如何針對某些必填引數丟擲異常

ES6這種新語法能夠讓我們很好的針對某個必填引數進行錯誤提醒:

function requireParams () {
  throw new Error(`required params`)
}

function doSomething (name = requireParams(), age = 18) {
  // do something
}

 

如果name引數為undefined,就會觸發預設值規則
然後呼叫requireParams函式,而我們在函式中直接throw了一個Error

複雜結構引數的預設值處理

上邊的處理都是針對簡單的基本型別資料進行處理的,但如果我們有如下的一個函式:

function init ({id, value}) {}

init({
  id: `tagId`,
  value: 1
})

 

如果在ES5環境下,針對這種引數的預設值處理將會變得無比複雜
首先要判斷這一個引數是否存在,然後在判斷引數中的所有key是否存在
而在ES6中,可以這樣來做:

function init ({
  id    = `defaultId`,
  value = 1
} = {}) {
  console.log(id, value)
}

init()

 

首先在解構函式的後邊新增預設值= {},然後針對每一項引數新增預設值,很簡潔的就實現了我們的需求。

ES5版本的polyfill程式碼在倉庫中的位置:defaultValue

參考資料

  1. MDN

相關文章