js 純函數(shù)

什么是純函數(shù)

純函數(shù)是函數(shù)式編程中非常重要的一個(gè)概念压汪,簡(jiǎn)單來(lái)說(shuō),就是一個(gè)函數(shù)的返回結(jié)果只依賴于它的參數(shù),并且在執(zhí)行過(guò)程中沒(méi)有副作用,我們就把這個(gè)函數(shù)叫做純函數(shù)

劃重點(diǎn):

  1. 函數(shù)的返回結(jié)果只依賴于它的參數(shù)
  2. 函數(shù)執(zhí)行過(guò)程中沒(méi)有副作用

首先來(lái)看第一點(diǎn):函數(shù)的返回結(jié)果只依賴于它的參數(shù)

const a = 1
const impure = (b)=>a + b
impure(2) // 3

上面代碼中热康, impure函數(shù)不是一個(gè)純函數(shù),因?yàn)樗姆祷亟Y(jié)果依賴于外部變量a, 因?yàn)閍是有可能變化的劣领, 所以我們不能保證 impure(2)的值永遠(yuǎn)是3. 雖然impure函數(shù)的代碼沒(méi)有變化姐军, 傳入的參數(shù)也沒(méi)有變化, 但它的返回值是不可預(yù)料的尖淘。 我們?cè)賮?lái)改一下:

const a = 1
const pure = (x, b) => x + b
pure(1,2) //3

現(xiàn)在庶弃, pure 滿足純函數(shù)的第一個(gè)條件,它的返回結(jié)果只依賴于它的參數(shù) x 和 b, 就是說(shuō)德澈, 只要代碼不變, pure(1,2)的返回值永遠(yuǎn)是3.

這就是純函數(shù)的第一個(gè)條件: 函數(shù)的返回結(jié)果只依賴于它的參數(shù)

接下來(lái)解釋第二點(diǎn):函數(shù)執(zhí)行中沒(méi)有副作用

副作用指的是: 在計(jì)算結(jié)果的過(guò)程中固惯,系統(tǒng)狀態(tài)的一種變化梆造, 或者與外部事件進(jìn)行觀察的交互。 再看一個(gè)例子:

var values = { a: 1 };
function impureFunction ( items ) {  
  var b = 1; 
  items.a = items.a * b + 2;  
  return items.a;
}
var c = impureFunction( values );
values.a // 3

在上面的代碼中葬毫,我們改變了參數(shù)對(duì)象中的屬性a, 由于我們定義的函數(shù)改變的對(duì)象在我們的函數(shù)作用域之外镇辉,導(dǎo)致這個(gè)函數(shù)稱為“不純”函數(shù)

var values = { a: 1 };
function pureFunction ( a ) {  
  var b = 1;  
  a = a * b + 2;  
  return a;
}
var c = pureFunction( values.a );
values.a // 1

上面的代碼,我們只計(jì)算了作用域內(nèi)的局部變量贴捡, 沒(méi)有任何作用域外部的變量被改變忽肛, 因此這個(gè)函數(shù)是 “純函數(shù)”

除了修改外部的變量, 一個(gè)函數(shù)在執(zhí)行過(guò)程中還有很多方式產(chǎn)生外部可觀察的變化烂斋,比如說(shuō)調(diào)用 DOM API修改頁(yè)面屹逛, 或者你發(fā)送了Ajax請(qǐng)求, 還有調(diào)用window.reload刷新瀏覽器汛骂, 甚至是console.log往控制臺(tái)打印數(shù)據(jù)也是副作用罕模。

總結(jié)一些副作用:

  1. 更改輸入
  2. console.log
  3. HTTP請(qǐng)求(AJAX,fetch)
  4. 更改文件系統(tǒng)
  5. DOM查詢

純函數(shù)很嚴(yán)格, 也就是說(shuō)你幾乎除了計(jì)算數(shù)據(jù)以外什么都不能干帘瞭, 計(jì)算的時(shí)候還不能依賴除了函數(shù)參數(shù)以外的數(shù)據(jù)淑掌。

我們常用的JS中的兩個(gè)方法: splice和slice來(lái)舉一個(gè)例子

var array1 = [0,1,2,3,4,5,6];
var array2 = [0,1,2,3,4,5,6];
var spliceArray = array1.splice(0,2);
var sliceArray = array2.slice(0,2);
console.log('array1: ' + array1);
console.log('spliceArray: ' + spliceArray);
console.log('array2: ' + array2);
console.log('sliceArray: ' + sliceArray); 

運(yùn)行結(jié)果:

array1: 2,3,4,5,6
spliceArray: 0,1
array2: 0,1,2,3,4,5,6
sliceArray: 0,1

可以看到, slice和splice的作用是大致相同的蝶念, 但是slice改變了原數(shù)組抛腕, 而slice卻沒(méi)有, 實(shí)際開(kāi)發(fā)中媒殉, slice這種不改變?cè)瓟?shù)組的方式更安全一些担敌,改變?cè)瓟?shù)組,是一種副作用

非純函數(shù)帶來(lái)的副作用
function getName(obj){    
  return obj.name;
}
function getAge(obj){  
  return obj.age;
}
function sayHi(person){  
  console.log('I am' + getName(person) + ',and I am' + getAge(person) + 'years old');
}
var Tom = {  name: 'TOM',  age: 26};
sayHi(Tom);

這里的sayHi不是純函數(shù)廷蓉, 它依賴于 getName柄错, getAge兩個(gè)函數(shù), 如果我不小心改變了其中某個(gè)函數(shù)的功能, 這將使得sayHi中國(guó)函數(shù)出現(xiàn)錯(cuò)誤售貌。 當(dāng)代碼變得復(fù)雜给猾,且由多人維護(hù)的時(shí)候,bug調(diào)試會(huì)變得非常復(fù)雜

使用純函數(shù)的優(yōu)點(diǎn)

1. 可復(fù)用性
純函數(shù)僅依賴于傳入的參數(shù)颂跨,這意味著你可以隨意將這個(gè)函數(shù)移植到別的代碼中敢伸, 只需要提供它需要的參數(shù)即可。 如果是非純函數(shù)恒削, 有可能你需要一根香蕉池颈,卻需要將整個(gè)香蕉樹(shù)都搬過(guò)去

2. 可測(cè)試性
純函數(shù)非常容易進(jìn)行單元測(cè)試,因?yàn)椴恍枰紤]上下文環(huán)境钓丰, 只需要考慮輸入和輸出

3. 并行代碼
純代碼是健壯的躯砰, 改變執(zhí)行次序不會(huì)對(duì)系統(tǒng)造成影響, 因此純函數(shù)的操作可以并行執(zhí)行携丁。

總結(jié)

  • 不產(chǎn)生副作用琢歇, 并且相同的輸入總是返回相同的輸出,那么這個(gè)函數(shù)是純函數(shù)梦鉴。
  • 副作用包括但不限于: 更改輸入李茫,HTTP請(qǐng)求, 磁盤(pán)寫(xiě)入肥橙,打印
  • 可以將函數(shù)輸入拷貝一份用于更改魄宏,不要去動(dòng)原來(lái)的數(shù)據(jù)。
  • 擴(kuò)展運(yùn)算符語(yǔ)法(...語(yǔ)法)是一個(gè)復(fù)制對(duì)象或者數(shù)據(jù)的簡(jiǎn)便方法存筏。

雖然純函數(shù)有很多優(yōu)點(diǎn)宠互, 但也要避免濫用的情況,函數(shù)越純椭坚,對(duì)環(huán)境依賴越小名秀, 往往意味著要傳入更多的參數(shù)。 我們的最終目的是: 讓你的代碼盡可能簡(jiǎn)單易懂和靈活藕溅。盡量在合適的場(chǎng)景使用它匕得。

參考:https://juejin.im/post/5a92a12f6fb9a063543c4c2b

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市巾表,隨后出現(xiàn)的幾起案子汁掠,更是在濱河造成了極大的恐慌,老刑警劉巖集币,帶你破解...
    沈念sama閱讀 217,907評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件考阱,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡鞠苟,警方通過(guò)查閱死者的電腦和手機(jī)乞榨,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,987評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)秽之,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人吃既,你說(shuō)我怎么就攤上這事考榨。” “怎么了鹦倚?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,298評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵河质,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我震叙,道長(zhǎng)掀鹅,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,586評(píng)論 1 293
  • 正文 為了忘掉前任媒楼,我火速辦了婚禮乐尊,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘划址。我一直安慰自己扔嵌,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,633評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布猴鲫。 她就那樣靜靜地躺著,像睡著了一般谣殊。 火紅的嫁衣襯著肌膚如雪拂共。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,488評(píng)論 1 302
  • 那天姻几,我揣著相機(jī)與錄音宜狐,去河邊找鬼。 笑死蛇捌,一個(gè)胖子當(dāng)著我的面吹牛抚恒,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播络拌,決...
    沈念sama閱讀 40,275評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼俭驮,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了春贸?” 一聲冷哼從身側(cè)響起混萝,我...
    開(kāi)封第一講書(shū)人閱讀 39,176評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎萍恕,沒(méi)想到半個(gè)月后逸嘀,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,619評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡允粤,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,819評(píng)論 3 336
  • 正文 我和宋清朗相戀三年崭倘,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了翼岁。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,932評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡司光,死狀恐怖琅坡,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情飘庄,我是刑警寧澤脑蠕,帶...
    沈念sama閱讀 35,655評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站跪削,受9級(jí)特大地震影響谴仙,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜碾盐,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,265評(píng)論 3 329
  • 文/蒙蒙 一晃跺、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧毫玖,春花似錦掀虎、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,871評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至阐滩,卻和暖如春二打,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背掂榔。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,994評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工继效, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人装获。 一個(gè)月前我還...
    沈念sama閱讀 48,095評(píng)論 3 370
  • 正文 我出身青樓瑞信,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親穴豫。 傳聞我的和親對(duì)象是個(gè)殘疾皇子凡简,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,884評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 純函數(shù)簡(jiǎn)單定義 1.函數(shù)返回的結(jié)果只依賴它的參數(shù)2.函數(shù)的執(zhí)行不會(huì)對(duì)該函數(shù)之外的其他對(duì)象等造成影響(副作用) 舉例...
    vinterx閱讀 202評(píng)論 0 0
  • 純函數(shù) 若一個(gè)函數(shù)對(duì)相同的輸入,永遠(yuǎn)會(huì)得到相同的輸出精肃,并且不會(huì)影響該函數(shù)作用域以外的環(huán)境變量潘鲫,則此函數(shù)稱為純函數(shù)。...
    PorcoPP閱讀 2,203評(píng)論 0 6
  • //Clojure入門(mén)教程: Clojure – Functional Programming for the J...
    葡萄喃喃囈語(yǔ)閱讀 3,662評(píng)論 0 7
  • 這本書(shū)是由臺(tái)灣作家林奕含所寫(xiě),這既是他的處女作又是她的絕命之作状植。說(shuō)這本書(shū)是他的群名制作是因?yàn)檫@本書(shū)寫(xiě)完發(fā)行沒(méi)有多久...
    耋紫閱讀 614評(píng)論 1 1
  • 樂(lè)乙閱讀 152評(píng)論 0 0