函数式编程之-bind函数
Bind函數(shù)
Bind函數(shù)在函數(shù)式編程中是如此重要,以至于函數(shù)式編程語言會為bind函數(shù)設(shè)計語法糖。另一個角度Bind函數(shù)非常難以理解,幾乎很少有人能通過簡單的描述說明白bind函數(shù)的由來及原理。
這篇文章試圖通過“人話”來描述bind函數(shù),并通過淺顯的實例為零函數(shù)式編程語言的開發(fā)者揭秘bind函數(shù)的作用及用法。
你一定寫過類似的代碼,估計你也明白這樣的代碼看起來很丑陋,一層層的判空嵌套打亂了代碼的主題結(jié)構(gòu)。
有沒法讓他變的更優(yōu)雅?當然你可以通過"early return"的做法,不過這種方式不在我們的討論范圍之內(nèi)。
這種風格的代碼存在一個明顯的code smell, GetFirstThing()/GetSecondThing()/GetThirdThing()等方法有可能返回null,我們說return null是一種不真確的做法,相關(guān)分析見拒絕空引用異常。使用Optional類型重構(gòu)如下:
看起來代碼結(jié)果跟之前一模一樣,重構(gòu)后的代碼并沒有變得更漂亮。不過現(xiàn)在的GetFirstThing()/GetSecondThing()/GetThirdThing()方法返回值為Optional<string>類型,不再是普通的string類型:
public Optional<string> GetFirstThing(int id) {//...return Optional.None<string>(); }重構(gòu)后的這段代碼很有意思,我們可以從函數(shù)組合的角度來讓整個代碼段變的更加優(yōu)雅。
組合
這段代碼其實做了一件事,那就是通過調(diào)用三個函數(shù)GetFirstThing()/GetSecondThing()/GetThirdThing()來完成一個業(yè)務(wù)邏輯。從函數(shù)式編程思想的角度出發(fā),我們傾向于把若干個小的函數(shù)連接起來,根據(jù)以前學過的知識,只有一個輸入和一個輸出的函數(shù)才能連接起來:
他們之所以能夠連接是因為這兩個函數(shù)的簽名一致,都擁有一個輸入和一個輸出。
例如:int -> string, string -> bool就可以組合為int -> bool。
而我們此時擁有的三個函數(shù)方法簽名如下:
顯然GetFirstThing和GetSecondThing是無法直接連接的,原因是GetFirstThing返回了Optional<string>類型,而GetSecondThing的輸入?yún)s是一個普通的string類型。如果我們能夠在Optional<T>上擴展一個函數(shù),函數(shù)接受一個簽名為T -> Optional<T>的函數(shù),那么我們就有可能將上面的三個函數(shù)串聯(lián)起來:
public static class Optional {public static Optional<T> Bind<T>(this Optional<T> input, Func<T, Optional<T>> f){if (input.HasValue()){return f(input.Value);}return Optional.None<T>();} }有了上面這個神奇的bind函數(shù)你就可以將上面的三個函數(shù)連接起來了:
public string GetSomething(int id) {return GetFirstThing(id).Bind(GetSecondThing).Bind(GetThirdThing); }用F#實現(xiàn):
let address = getFirstThing id|> bind getSecondThing|> bind getThirdThing通過bind函數(shù)我們成功將三個函數(shù)連接了起來, 同時將判空放在了bind函數(shù)里,從而保持主要邏輯部分更加線性和清晰。
如何編寫屬于自己的bind函數(shù)
List<T>中的bind函數(shù)
我們經(jīng)常用的List<T>就是一個典型的泛型類型,那他上面有沒有bind函數(shù)?當然有,不過叫做SelectMany, Scala中也叫flatMap。
看一下SelectMany的方法簽名,正好符合bind函數(shù)的簽名要求:
SelectMany可以用在什么樣的場景中?
例如有這樣一個場景,一篇文章(paper)可以有若干章節(jié)(section)組成,每個章節(jié)(section)又有若干行(row)組成,每行(row)有若干單詞(word)組成。
問:給定一篇文章(paper),請找出大于10行(row)的章節(jié)(section),里面排除注釋的行(row)總共的單詞(word)數(shù)量。
首先根據(jù)需求變下下面的若干函數(shù):
private List<Paper.Section> GetSections(Paper paper) {return paper.Sections.Where(s => s.Rows.Count > 10).ToList(); }private List<Paper.Section.Row> GetRows(Paper.Section section) {return section.Rows.Where(r=>!r.IsComment).ToList(); }private List<Paper.Section.Row.Word> GetWords(Paper.Section.Row row) {return row.Words; }且看這三個函數(shù)的簽名:
GetSections: Papaer -> List<Section> GetRows: Section -> List<Row> GetWords: Row -> List<Word>正好這就是就符合bind函數(shù)連接的需求:
var length = GetSections(paper).SelectMany(GetRows).SelectMany(GetWords).Count();F#實現(xiàn):
let words = getSections paper|> bind getRows|> bind getWordswords.Lengthbind函數(shù)的語法糖支持
bind函數(shù)在函數(shù)式編程中如此常見,以至于需要設(shè)計單獨的語法糖,Haskell中叫do natation, Scala中叫for comprehension,F#用Computation expressions:
list {let! section = getSections(paper)let! row = getRows(section)let! word = getWord(row)return word }轉(zhuǎn)載于:https://www.cnblogs.com/xiandnc/p/9684271.html
總結(jié)
以上是生活随笔為你收集整理的函数式编程之-bind函数的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 关闭360浏览器广告(广告洁癖)
- 下一篇: WPF里ItemsControl的分组实