如何在ES5與ES6環境下處理函數默認參數

函數默認值是一個很提升魯棒性的東西(就是讓程序更健壯)
MDN關於函數默認參數的描述:函數默認參數容許在沒有值或 undefined被傳入時使用默認形參。

ES5

使用邏輯或||來實現

衆所周知,在ES5版本中,並無提供的直接方法供咱們咱們處理函數默認值
因此只可以本身去加強函數的功能,通常會這麼來作:javascript

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

  console.log(name, age)
}

咱們將函數的兩個參數nameage進行默認值的處理,若是沒有則使用默認值。
在執行一下函數後,好像並無什麼不對:java

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

然而當咱們執行這樣的代碼時,就會得到一些超出預期的結果:git

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

可以發現,對於參數0,咱們上邊的默認參數實現方法是有問題的github

就像下邊的四個表達式,都會輸出wrong,這很顯然不可以知足上邊MDN關於函數默認參數的定義:函數

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

正確的姿式

因此,在ES5中正確的默認值處理應該是這樣:ui

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

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

  console.log(name, age)
}

使用三元運算符簡化操做

或者咱們簡寫成三元運算符形式的:es5

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

  console.log(name, age)
}

使用函數進行封裝

可是若是咱們每寫一個函數,都要重複的去作這些操做
未免太麻煩了,因此,咱們對這個邏輯進行一個簡單的封裝:code

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實現了函數默認參數的邏輯ip

one momre things

關於上邊的defaultValue函數實現方法,咱們在合理的使用弱類型語言的優點後
能夠使用這種方式來省去三元運算符的操做:get

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
相關文章
相關標籤/搜索