學習筆記(一)—— 函數(shù)式編程

最近開始對前端知識體系進行一個系統(tǒng)的擴展跟學習怒允,通過每天定期的學習,對一些不常使用的知識點進行了解和補充锈遥,同時對已經(jīng)使用過的知識點溫故而知新
在此記錄學習筆記纫事,并根據(jù)學習進度定時跟新

函數(shù)一等公民 (First-class Function)

  • 函數(shù)可以存儲在變量中
  • 函數(shù)可以作為參數(shù)
  • 函數(shù)可以作為返回值

高階函數(shù)(Higher-order Function)

  • 可以把函數(shù)作為參數(shù)傳遞給另一個函數(shù)
  • 可以把函數(shù)作為另一個函數(shù)的返回結(jié)果

常用的高階函數(shù)及模擬實現(xiàn)

  • forEach
// forEach
const forEach = (array, fn) => {
    for (let t of array) {
        fn(t)
    }
}
  • filter
// filter
const filter = (array, fn) => {
    let result = []
    for (let t of array) {
        fn(t) && result.push(t)
    }
    return result;
}
  • map
// map
const map = (array, fn) => {
    let result = []
    for (let t of array) {
        result.push(fn(t))
    }
    return result
}
  • some
// some
const some = (array, fn) => {
    for (let t of array) {
        if(fn(t)) {
            return true
        }
    }
    return false
}
  • every
// every
const every = (array, fn) => {
    for (let t of array) {
        if(!fn(t)) {
            return false
        }
    }
    return true
}
  • once
// once
const once = (fn) => {
    let done = false
    return (...args) => {
        if (!done) {
            done = true
            return fn.apply(this, args)
        }
    }
}

閉包 (Closure)

  • 函數(shù)及其周圍的狀態(tài)的引用捆綁在一起勘畔,組成閉包
    * 可以在另一個作用域中調(diào)用一個函數(shù)的內(nèi)部函數(shù),并訪問到該函數(shù)的作用域中的成員

  • 閉包的本質(zhì):函數(shù)在執(zhí)行時會放到一個執(zhí)行棧上丽惶,當執(zhí)行完成后會從執(zhí)行棧上移除炫七,但是堆上的作用域成員因為被外部引用而不能被釋放,因此內(nèi)部函數(shù)依然可以訪問外部函數(shù)的成員

  • 箭頭函數(shù)蚊夫,閉包中this指向undefined诉字,因為外層函數(shù)調(diào)用棧被釋放?

純函數(shù)

  • 相同的輸入永遠得到相同的輸出,且沒有任何可觀察的副作用
  • 數(shù)組的slice和splice分別是純函數(shù)和不純函數(shù)
    * slice返回數(shù)組指定部分知纷,不會改變原數(shù)組
    * splice從原數(shù)組中移除的元素壤圃,并返回移除元素集合,會改變原數(shù)組
  • 函數(shù)式編程不會保留計算中間的結(jié)果琅轧,所以變量是不可變的(無狀態(tài)的)

Lodash

具體參見Lodash官網(wǎng)

純函數(shù)的好處

  • 可緩存

    • 模擬實現(xiàn)lodash memoize
    function memoize (fn) {
        let cache = {}
        return function () {
            const key = JSON.stringify(arguments)
            cache[key] = cache[key] || fn(...arguments)
            return cache[key]        
        }
    }
    
  • 可測試
  • 并行處理
  • 純函數(shù)不需要訪問共享內(nèi)存數(shù)據(jù)

副作用

  • 使純函數(shù)變得不純
  • 副作用來源:外部數(shù)據(jù)依賴(接口伍绳、數(shù)據(jù)庫、配置文件乍桂、全局變量等)
  • 副作用不可能完全禁止冲杀,盡可能控制在可控范圍內(nèi)

柯里化(Currying)

  • 當一個函數(shù)包含多個參數(shù)的時候先傳遞部分參數(shù)去調(diào)用他(這部分參數(shù)以后永遠不變)
  • 然后返回一個新的函數(shù)接收剩余參數(shù),并返回結(jié)果
  • 柯里化模擬實現(xiàn)
// curry
function curry (fn) {
    return function (...args) {
        if (args.length < fn.length) {
            return function() {
                return fn(...args, ...arguments)
            }
        }
        return fn(...args)
    }
}
  • 柯里化總結(jié)
    • 柯里化可以使我們給一個函數(shù)傳遞較少的參數(shù)睹酌,生成一個記住了某些參數(shù)的新函數(shù)
    • 是一種對函數(shù)的緩存
    • 使函數(shù)的粒度更小
    • 可以將多元(多參數(shù))函數(shù)轉(zhuǎn)化為一元(單參數(shù))函數(shù)

函數(shù)組合(compose)

  • 函數(shù)組合可以把細粒度的函數(shù)重新組合成新的函數(shù)

  • 如果一個函數(shù)需要多個函數(shù)處理才能得到最終值权谁,我們可以將中間過程的函數(shù)合并成一個函數(shù)

  • 函數(shù)組合默認從右向左執(zhí)行

  • compose函數(shù)模擬實現(xiàn)

    // compose
    const compose = (...fns) => (...args) => fns.reduceRight((fn1, fn2) => ()=> fn2(fn1(...args)))()
    
  • lodash中的組合函數(shù)

    • flow() 從左向右組合執(zhí)行
    • flowRight() 從右向左組合執(zhí)行
  • lodash/fp模塊

    • 提供了對函數(shù)式編程友好的方法
    • fp模塊中提供的方法(例如:map、filter憋沿、split等)都是被柯里化的
    • 多個參數(shù)的方法函數(shù)優(yōu)先旺芽,數(shù)據(jù)滯后(默認為數(shù)據(jù)優(yōu)先、函數(shù)滯后辐啄,例如map(array, func))
  • lodash-map方法的小問題:parseInt參數(shù)問題

    • map(array, func(item, index, array))
    • parseInt(string, radix)
      • string 待parse字符串
      • radix 進制
    • map傳遞給parseInt 3個參數(shù)采章,將index當成了radix,由此引發(fā)問題
    • fp.map柯里化后的func只接收一個參數(shù)壶辜,因此沒有此問題
  • 函數(shù)組合只能組合單參數(shù)函數(shù)悯舟?

  • PointFree

    • 不需要指明處理的數(shù)據(jù)
    • 只需要合成運算過程
    • 需要定義一些輔助的基本運算函數(shù)

函子(Functor)

  • 容器:包含值和值的變形關(guān)系(變形關(guān)系即函數(shù))

  • 函子:是一個特殊的容器,通過一個普通對象來實現(xiàn)砸民,該對象具有map方法抵怎,map方法可以運行一個函數(shù)對值進行變形處理

  • 函子用來處理副作用、異步操作岭参、異常處理等

  • 函子對象class實現(xiàn)

    class Functor {
        constructor(value) {
            this._value = value
        }
        static of(value) {
            return new this.prototype.constructor(value)
        }
        map(fn) {
            return this.of(fn(this._value))
        }
    }
    
  • 總結(jié)

    • 函數(shù)式編程的運算不直接操作值便贵,而由函子完成
    • 函子就是一個實現(xiàn)了map契約的對象
    • 函子中封裝了一個值,想要操作值冗荸,就給map傳遞一個處理值的函數(shù)(純函數(shù))
    • 最終返回一個新的函子
  • 函子介紹

    • MayBe函子
      • 用于處理空值(null、undefined)
    class MayBe extends Functor {
        isNothing() {
            return this._value === null || this._value === undefined
        }
        map(fn) {
            return this.isNothing() ? MayBe.of(null) : MayBe.of(fn(this._value))
        }
    }
    
    • Either函子
    • 用于處理異常
    • 類似于if...else
    • Left函子 + Right函子
    • IO函子
      • IO函子的value是一個函數(shù)利耍,將函數(shù)作為值來處理
      • IO函子可以把不純的動作存儲到_value中蚌本,延遲執(zhí)行(惰性執(zhí)行)
    class IO extends Functor {
        static of(value) {
            return new IO(function() {
                return value
            })
        }
        map(fn) {
            return new IO(fp.flowRight(fn, this._value))
        }
    }
    
    • Task函子
      • 用于處理異步任務
    const { task } = require('folktale/concurrency/task')
    const { split, find } = require('lodash/fp')
    const fs = require('fs')
    
    function readFile(filename) {
        return task(resolver => {
            fs.readFile(filename, 'utf-8', (err, data) => {
                if(err) {
                    resolver.reject(err)
                }
                resolver.resolve(data)
            })
        })
    }
    
    readFile('package.json')
        .map(split('\n'))
        .map(find(x => x.includes('version')))
        .run()
        .listen({
            onRejected: e => console.log(e),
            onResolved: d => console.log(d),
        })
    
    • Pointed函子
      • 實現(xiàn)了of靜態(tài)方法的函子
      • of靜態(tài)方法避免了使用new 來創(chuàng)建對象
      • 更深層的含義是of方法用來把值放到上下文Context(把值放到容器中盔粹,使用map來處理)
    • Monad函子
      • 一個函子具有join和of兩個方法,并遵守一定規(guī)律程癌,就是Monad函子
      • 可以變扁的Pointed函子
  • folktale 一個標準的函數(shù)式編程庫

    • 提供了compose舷嗡、curry等函數(shù)
    • 提供了Task、Maybe嵌莉、Either等函子
    • 用法自行百度
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末进萄,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子锐峭,更是在濱河造成了極大的恐慌中鼠,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,686評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件沿癞,死亡現(xiàn)場離奇詭異援雇,居然都是意外死亡,警方通過查閱死者的電腦和手機椎扬,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,668評論 3 385
  • 文/潘曉璐 我一進店門惫搏,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人蚕涤,你說我怎么就攤上這事筐赔。” “怎么了揖铜?”我有些...
    開封第一講書人閱讀 158,160評論 0 348
  • 文/不壞的土叔 我叫張陵茴丰,是天一觀的道長。 經(jīng)常有香客問我蛮位,道長较沪,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,736評論 1 284
  • 正文 為了忘掉前任失仁,我火速辦了婚禮尸曼,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘萄焦。我一直安慰自己控轿,他們只是感情好,可當我...
    茶點故事閱讀 65,847評論 6 386
  • 文/花漫 我一把揭開白布拂封。 她就那樣靜靜地躺著茬射,像睡著了一般。 火紅的嫁衣襯著肌膚如雪冒签。 梳的紋絲不亂的頭發(fā)上在抛,一...
    開封第一講書人閱讀 50,043評論 1 291
  • 那天,我揣著相機與錄音萧恕,去河邊找鬼刚梭。 笑死肠阱,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的朴读。 我是一名探鬼主播屹徘,決...
    沈念sama閱讀 39,129評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼衅金!你這毒婦竟也來了噪伊?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,872評論 0 268
  • 序言:老撾萬榮一對情侶失蹤氮唯,失蹤者是張志新(化名)和其女友劉穎鉴吹,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體您觉,經(jīng)...
    沈念sama閱讀 44,318評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡拙寡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,645評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了琳水。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片肆糕。...
    茶點故事閱讀 38,777評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖在孝,靈堂內(nèi)的尸體忽然破棺而出诚啃,到底是詐尸還是另有隱情,我是刑警寧澤私沮,帶...
    沈念sama閱讀 34,470評論 4 333
  • 正文 年R本政府宣布始赎,位于F島的核電站,受9級特大地震影響仔燕,放射性物質(zhì)發(fā)生泄漏造垛。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 40,126評論 3 317
  • 文/蒙蒙 一晰搀、第九天 我趴在偏房一處隱蔽的房頂上張望五辽。 院中可真熱鬧,春花似錦外恕、人聲如沸杆逗。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,861評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽罪郊。三九已至,卻和暖如春尚洽,著一層夾襖步出監(jiān)牢的瞬間悔橄,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,095評論 1 267
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留橄维,地道東北人尺铣。 一個月前我還...
    沈念sama閱讀 46,589評論 2 362
  • 正文 我出身青樓,卻偏偏與公主長得像争舞,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子澈灼,可洞房花燭夜當晚...
    茶點故事閱讀 43,687評論 2 351