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

  • 2019 年 12 月 9 日
  • 筆記

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

函數默認值是一個很提高魯棒性的東西(就是讓程序更健壯) 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