函數(shù)式內(nèi)功心法-01: 流程控制神器之callcc浪子回頭

本人是一個(gè)函數(shù)式愛好者验烧,苦于網(wǎng)上資料貧乏以及術(shù)語理論性太強(qiáng)久矣持舆。
故本人決定用白話文講述函數(shù)式技術(shù)色瘩,當(dāng)然會(huì)不那么準(zhǔn)確,但是便于理解其基本概念逸寓。

在函數(shù)式世界里,每個(gè)函數(shù)語句都要有返回值竹伸。

一個(gè)函數(shù)里面進(jìn)行各種操作的時(shí)候 泥栖,怎樣能夠在特定情況下提前返回呢?
當(dāng)然在面向過程語言里面勋篓,比較相似的就是return語句了吧享。
比如下面的例子:

whatsYourName :: String -> String
whatsYourName name =
  (`runCont` id) $ do                      -- 1
    response <- callCC $ \exit -> do       -- 2
      validateName name exit               -- 3
      return $ "Welcome, " ++ name ++ "!"  -- 4
    return response                        -- 5

validateName name exit = do
  when (null name) (exit "You forgot to tell me your name!")

對(duì)名稱進(jìn)行驗(yàn)證,驗(yàn)證失敗后則直接返回譬嚣。

這里在函數(shù)式是怎么做到的呢钢颂? 居然直接忽略后面的語句提前結(jié)束!

在函數(shù)式世界里面,正常情況下是不可能發(fā)生的孤荣。
除非我們有多通道甸陌,正常走正常的通道须揣,特殊走VIP通道。
沒說钱豁,我們就是可以這么干耻卡!

1. 首先我們引入CPS(continuation passing style)的概念。

什么是CPS牲尺?CPS就像一場(chǎng)接力比賽卵酪。
來看看簡(jiǎn)單的例子: 1 * 2 + 3 * 4
首先我們計(jì)算 1 * 2,然后計(jì)算3 * 4, 最后累加.
這里就涉及三次傳遞過程谤碳。
1 * 2 -> 3 * 4 -> a1 + a2 -> ?
每一次結(jié)果都往后傳遞處理溃卡,帶上自己一直傳遞下去,這就是CPS風(fēng)格蜒简。
誰來接最后一棒呢瘸羡? 最后一棒有個(gè)特殊的名字,叫做終級(jí)continuation搓茬。

2. 我們?cè)賮砜辞懊娴膯栴}

假如我們共有五次接力犹赖。假設(shè)第二次接力可能出現(xiàn)問題。
我們對(duì)第二次接力進(jìn)行驗(yàn)證卷仑,如果沒有問題峻村,則繼續(xù)往下接力。
如果有問題锡凝,直接找終級(jí)ccontinuation最后一棒粘昨!
所以問題似乎很簡(jiǎn)單了。窜锯。张肾。

既然思想通了,那么就該開始練習(xí)內(nèi)功心法了衬浑!

這里涉及haskell的兩個(gè)庫: transformer以及mtl捌浩。
mtl在transformer上對(duì)monad transformer做了增強(qiáng)。
mtl上面有個(gè)Control.Monad.Cont提供了callcc接口工秩,用于實(shí)現(xiàn)VIP通道功能尸饺。
底層均由transformer的Control.Monad.Trans.Cont實(shí)現(xiàn)

相關(guān)源碼如下:
mtl: https://github.com/haskell/mtl/blob/master/Control/Monad/Cont/Class.hs
transformer: https://hub.darcs.net/ross/transformers/browse/Control/Monad/Trans/Cont.hs

mtl本質(zhì)上沒干啥活,主義功能就是定義了一個(gè)MonadCont的接口(即typeclass)助币。其它的就是實(shí)現(xiàn)了底層Cont庫的MonadReader跟MonadState接口浪听。
所以,我們主要看transformer庫眉菱。

1. 首先我們定義傳遞迹栓,這個(gè)傳遞主要由ContT定義

newtype ContT r m a = ContT { runContT :: (a -> m r) -> m r }

這里定義了ContT類型,主要有三個(gè)類型變量r, m, a
a是當(dāng)前的傳遞值俭缓, m是對(duì)返回結(jié)果r進(jìn)行隔離的另一層數(shù)據(jù)結(jié)構(gòu)
整個(gè)過程就是對(duì)于已有的傳遞值a克伊, 等待一個(gè)傳遞函數(shù)a->m r酥郭,最終生成結(jié)果m r。
對(duì)于函數(shù)類型來說愿吹,輸入?yún)?shù)為等待值不从。這里等待a-> m r傳遞函數(shù)。
通過將自己的傳遞值a移交給傳遞函數(shù)犁跪,完成了傳遞功能 椿息。
這里的ConT僅僅是一種函數(shù)類型封裝,使用過程中則需要拆解以及再次封裝過程坷衍。

2. 傳遞是如何組合的呢?

    m >>= k  = ContT $ \ c -> runContT m (\ x -> runContT (k x) c)
  1. 首先m傳遞\x值后寝优,運(yùn)行下一棒傳遞函數(shù)runContT (k x) c
  2. k綁定m的返回結(jié)果x后生成新的ContT,繼續(xù)等待參數(shù)c傳遞函數(shù)
    因此枫耳,整 個(gè)過程比較簡(jiǎn)單乏矾,就是把當(dāng)前結(jié)果\x傳遞給下一次調(diào)用(k x)后, 生成新的ConT繼續(xù)等待終極傳遞函數(shù)。

3. 那么Cont又是如何構(gòu)造的呢?

看一個(gè)簡(jiǎn)單的例子:
參見https://en.wikipedia.org/wiki/Continuation-passing_style

pow2_m :: Float -> Cont a Float
pow2_m a = return (a ** 2)

add' :: Float -> Float -> (Float -> a) -> a
add' a b cont = cont (a + b)

sqrt' :: Float -> ((Float -> a) -> a)
sqrt' a = \cont -> cont (sqrt a)

pyth_m :: Float -> Float -> Cont a Float
pyth_m a b = do
  a2 <- pow2_m a
  b2 <- pow2_m b
  anb <- cont (add' a2 b2)
  r <- cont (sqrt' anb)
  return r

兩種Cont構(gòu)造:

instance Monad (ContT r m) where
    return x = ContT ($ x)

cont :: ((a -> r) -> r) -> Cont r a
cont f = ContT (\ c -> Identity (f (runIdentity . c)))

pow2_m :: Float -> Cont a Float
pow2_m a = return (a ** 2)
pow2' :: Float -> (Float -> a) -> a
pow2' a cont = cont (a ** 2)
  1. 通過return構(gòu)造, 調(diào)用$等待傳遞函數(shù)即得Cont Monad
  2. 通過cont函數(shù)調(diào)用嘉涌,將a -> m r傳遞函數(shù)作為參數(shù)調(diào)用并進(jìn)行等待

4. 傳遞函數(shù)如何傳遞呢?

newtype ContT r m a = ContT { runContT :: (a -> m r) -> m r }

runCont
    :: Cont r a         -- ^ continuation computation (@Cont@).
    -> (a -> r)         -- ^ the final continuation, which produces
                        -- the final result (often 'id').
    -> r
runCont m k = runIdentity (runContT m (Identity . k))

evalContT :: (Monad m) => ContT r m r -> m r
evalContT m = runContT m return

evalCont :: Cont r r -> r
evalCont m = runIdentity (evalContT m)

主要分為兩種妻熊,
一種是runCount系列,就是接受一個(gè)終極傳遞函數(shù)即可
另一種是evalCont系列仑最,即是將最后傳遞的值返回

callcc登場(chǎng)

前面的基礎(chǔ)知識(shí)有了,讓我們重新來回顧一下callcc的過程帆喇。

whatsYourName :: String -> String
whatsYourName name =
  (`runCont` id) $ do                      -- 1
    response <- callCC $ \exit -> do       -- 2
      validateName name exit               -- 3
      return $ "Welcome, " ++ name ++ "!"  -- 4
    return response                        -- 5

validateName name exit = do
  when (null name) (exit "You forgot to tell me your name!")

先看一下callcc是如何實(shí)現(xiàn)的

callCC :: ((a -> ContT r m b) -> ContT r m a) -> ContT r m a
callCC f = ContT $ \ c -> runContT (f (\ x -> ContT $ \ _ -> c x)) c
  1. 首先通過runCount運(yùn)行一個(gè)callcc Cont, 并使用id作為終極傳遞函數(shù)返回Cont的傳遞值
  2. callcc調(diào)用一個(gè)函數(shù)警医,傳遞VIP通道(a -> ContT r m b)逃逸Continuation作為函數(shù)參數(shù)
  3. 逃逸Continuation接受一個(gè)傳遞值,直接調(diào)用終極傳遞函數(shù)后結(jié)束傳遞坯钦。
    (\ x -> ContT $ \ _ -> c x)
  4. callcc函數(shù)里面接受逃逸continuation之后進(jìn)行CPS傳遞過程
  5. 正常邏輯一直通過monad組合傳遞下去预皇,特殊通道接受參數(shù)直接完成傳遞鏈
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市婉刀,隨后出現(xiàn)的幾起案子吟温,更是在濱河造成了極大的恐慌,老刑警劉巖突颊,帶你破解...
    沈念sama閱讀 218,546評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件鲁豪,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡律秃,警方通過查閱死者的電腦和手機(jī)爬橡,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,224評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來棒动,“玉大人糙申,你說我怎么就攤上這事〈遥” “怎么了柜裸?”我有些...
    開封第一講書人閱讀 164,911評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵缕陕,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我疙挺,道長(zhǎng)扛邑,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,737評(píng)論 1 294
  • 正文 為了忘掉前任衔统,我火速辦了婚禮鹿榜,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘锦爵。我一直安慰自己舱殿,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,753評(píng)論 6 392
  • 文/花漫 我一把揭開白布险掀。 她就那樣靜靜地躺著沪袭,像睡著了一般。 火紅的嫁衣襯著肌膚如雪樟氢。 梳的紋絲不亂的頭發(fā)上冈绊,一...
    開封第一講書人閱讀 51,598評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音埠啃,去河邊找鬼死宣。 笑死,一個(gè)胖子當(dāng)著我的面吹牛碴开,可吹牛的內(nèi)容都是我干的毅该。 我是一名探鬼主播,決...
    沈念sama閱讀 40,338評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼潦牛,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼眶掌!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起巴碗,我...
    開封第一講書人閱讀 39,249評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤朴爬,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后橡淆,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體召噩,經(jīng)...
    沈念sama閱讀 45,696評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,888評(píng)論 3 336
  • 正文 我和宋清朗相戀三年明垢,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了蚣常。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,013評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡痊银,死狀恐怖抵蚊,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤贞绳,帶...
    沈念sama閱讀 35,731評(píng)論 5 346
  • 正文 年R本政府宣布谷醉,位于F島的核電站,受9級(jí)特大地震影響冈闭,放射性物質(zhì)發(fā)生泄漏俱尼。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,348評(píng)論 3 330
  • 文/蒙蒙 一萎攒、第九天 我趴在偏房一處隱蔽的房頂上張望遇八。 院中可真熱鬧,春花似錦耍休、人聲如沸刃永。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,929評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽斯够。三九已至,卻和暖如春喧锦,著一層夾襖步出監(jiān)牢的瞬間读规,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,048評(píng)論 1 270
  • 我被黑心中介騙來泰國(guó)打工燃少, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留束亏,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,203評(píng)論 3 370
  • 正文 我出身青樓阵具,卻偏偏與公主長(zhǎng)得像枪汪,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子怔昨,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,960評(píng)論 2 355