您好,登錄后才能下訂單哦!
這篇文章主要講解了“怎么使用JS函數式編程實現XDM”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“怎么使用JS函數式編程實現XDM”吧!
函數式編程(FP),不是一個新的概念,它幾乎貫穿了整個編程史。直到最近幾年,函數式編程才成為整個開發界的主流觀念。
函數式編程有完善且清晰的原則,一旦我們知道這些原則,我們將能更加快速地讀懂代碼,定位問題。這是為什么函數式編程重要的原因!
比如:你可能寫過一些命令式的代碼,像 if 語句和 for 循環這樣的語句。這些語句旨在精確地指導計算機如何完成一件事情。而聲明式代碼,以及我們努力遵循函數式編程原則所寫出的代碼,更專注于描述最終的結果。
函數式編程以另一種方式來思考代碼應該如何組織才能使數據流更加明顯,并能讓讀者很快理解你的思想。
記住,你編寫的每一行代碼之后都要有人來維護,這個人可能是你的團隊成員,也可能是未來的你。
函數式編程不是僅僅用 function 這個關鍵詞來編程,就像面向對象編程不僅僅是用了對象就算是。
函數的真正意義是什么?
回到最初的起點,我們心中的函數一定是這樣的:
f(x) = 2x2 + 3,這是數學上真正的函數。那這和函數式編程有什么關系呢?
函數的本質是【映射】。以一個優雅的方式來描述一組值和另一組值的映射關系,即函數的輸入值與輸出值之間的關聯關系。
在編程中,它或許有許多個輸入值,或許沒有。它或許有一個輸出值( return 值),或許沒有。
如果你計劃使用函數式編程,你應該盡可能多地使用函數,而不是程序。你所有編寫的 function 應該接收輸入值,并且返回輸出值。(這么做的原因是多方面的,后續會一一介紹)
這里,輸入值就是函數傳參,輸出值就是return的東西。(如果你沒有 return 值,或者你使用 return;,那么則會隱式地返回 undefined 值。)
思考以下代碼:
function foo(x) { if (x > 10) return x + 1; var y = x / 2; if (y > 3) { if (x % 2 == 0) return x; } if (y > 1) return y; return x; }
foo(2) 返回什么? foo(4) 返回什么? foo(8), foo(12) 呢?
將上述代碼變形,請思考:
function foo(x) { var retValue; if (retValue == undefined && x > 10) { retValue = x + 1; } var y = x / 2; if (y > 3) { if (retValue == undefined && x % 2 == 0) { retValue = x; } } if (retValue == undefined && y > 1) { retValue = y; } if (retValue == undefined) { retValue = x; } return retValue; }
這樣寫會更容易理解嗎?
在每個 retValue 可以被設置的分支, 這里都有個守護者以確保 retValue 沒有被設置過才執行。(?)
相比在函數中提早使用 return,我們更應該用常用的流控制( if 邏輯 )來控制 retValue 的賦值。到最后,我們 return retValue。
用 return 來實現流控制,會創造更多的隱含意義。
再來思考以下代碼:
// 通過一個函數修改變量 y 的值 var y; function foo(x) { y = (2 * Math.pow( x, 2 )) + 3; } foo( 2 ); y;
但是我們也可以這樣寫:
function foo(x) { return (2 * Math.pow( x, 2 )) + 3; } var y = foo( 2 ); y;
后一個版本更有優勢嗎?
答案是肯定的:有!
后一個版本中的 return 表示一個顯式輸出,而前者的 y 賦值是一個隱式輸出。
通常,開發人員喜歡顯式模式而不是隱式模式。
為什么說后者 return 出來的就是顯式的?而前者的 y 賦值是隱式的?
這個例子可以給你答案:
function sum(list) { var total = 0; for (let i = 0; i < list.length; i++) { if (!list[i]) list[i] = 0; // list 使用了 nums 的引用,不是對 [1,3,9,..] 的值復制,而是引用復制。 total = total + list[i]; } return total; } var nums = [ 1, 3, 9, 27, , 84 ]; sum( nums ); // 124
這段代碼,除了 return 的輸出,還有沒有其它輸出可能改變到函數外部參數 nums 的值?
是有的!就是在注釋的一行,我們無意中改變了 nums 。
console.log(nums) // [1, 3, 9, 27, 0, 84]
JS 對數組、對象和函數都使用引用和引用復制,我們可以很容易地從函數中創建輸出,即使是無心的。
這個隱式函數輸出在函數式編程中有一個特殊的名稱:副作用。
沒有副作用的函數也有一個特殊的名稱:純函數,這個概念十分重要,后面對有更多討論!
一個函數如果可以接受或返回一個甚至多個函數,它被叫做高階函數。
其中最強大的就是:【閉包】。
我們將在的后續舉例中大量使用閉包。它可能是所有函數式編程中最重要的基礎。
此處舉一小例:
假設你需要將兩個值相加,一個你已經知道,另一個還需要后面才能知道,你可以使用閉包來記錄第一個輸入值:
function makeAdder(x) { return function sum(y){ return x + y; }; } //我們已經分別知道作為第一個輸入的 10 和 37 var addTo10 = makeAdder( 10 ); var addTo37 = makeAdder( 37 ); // 緊接著,我們指定第二個參數 addTo10( 3 ); // 13 addTo10( 90 ); // 100 addTo37( 13 ); // 50
這種在連續函數調用中指定輸入,是函數式編程中非常普遍的形式。
它可分為兩類:偏函數應用和柯里化。后續會展開。
我們提倡要用具名函數,而不是匿名函數,這更有利于我們語義化代碼,比如getPreferredName(..),操作意圖很明確,并且可以很好的回溯問題,防止出現 (anonymous function) 。
但是 => 箭頭函數除外,箭頭函數還是得有效利用。
=> 箭頭函數令人興奮的地方在于它幾乎完全遵循函數的數學符號,特別是像 Haskell 這樣的函數式編程語言。它能簡化、優化代碼片段中的空間。
JavaScript 中的 this 綁定規則是真的難記,好消息是我們將把 this 丟棄掉,不去理會它。
這樣做的內核原因是:this 是函數的一個隱式的輸入參數。前面我們提到通常,開發人員喜歡顯式模式而不是隱式模式。,這樣的隱式輸入違背了我們的原則。
感謝各位的閱讀,以上就是“怎么使用JS函數式編程實現XDM”的內容了,經過本文的學習后,相信大家對怎么使用JS函數式編程實現XDM這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。