以 Huffman coding 為例看函數(shù)式編程

不同編程即為不同解決問題的思路

解決一個問題有很多思路煮落,比如:

  1. 過程式(C語言):將解決問題的方式分解成若干步驟來控制數(shù)據(jù)澜建。
  2. 面向?qū)ο笫?(Java/Ruby):以對象為基本單位署恍,定義其行為控制其內(nèi)部狀態(tài)碧查,通過不同對象之間的協(xié)作解決問題约啊。
  3. 函數(shù)式(Lisp):一切皆為函數(shù)喻喳,用連貫的思維方式定義過程另玖,常用遞歸。
  4. 組合式:將不同的解決方式組合起來表伦,golang 經(jīng)常會將面向?qū)ο笈c過程式組合起來谦去。

示例: Huffman編碼

用 Lisp 的一個方言 Scheme 來實現(xiàn):

輸入 '((A 1) (B 3) (C 2)), A B C 為帶編碼字符,1 2 3 為出現(xiàn)次數(shù)蹦哼。
輸出 '((b 0) (a 0 1) c 1 1)

定義葉子節(jié)點

(define (make-leaf symbol weight)
  (list 'leaf symbol weight))

(define (leaf? object)
  (eq? (car object) 'leaf))

(define (symbol-leaf object)
  (cadr object))

(define (weight-leaf object)
  (caddr object))

如果沒有接觸過 lisp 的同學(xué)可能對上面的表示方式有點陌生鳄哭,其實就是用括號代表方法調(diào)用,括號里的第一個位置是方法名稱纲熏,后面的是調(diào)用該方法的參數(shù)妆丘。

上面的幾行代碼定義了葉子節(jié)點 leaf 及相關(guān)函數(shù)锄俄。

定義樹節(jié)點
(define (make-code-tree left right)
  (list 
    left
    right
    (append (symbols left) (symbols right))
    (+ (weight left) (weight right))))

(define (left-branch tree) (car tree))

(define (right-branch tree) (cadr tree))

(define (symbols tree)
  (if (leaf? tree)
    (list (symbol-leaf tree))
    (caddr tree)))

(define (weight tree)
  (if (leaf? tree)
    (weight-leaf tree)
    (cadddr tree)))

同樣的道理,定義了用于在構(gòu)造 Huffman 樹中非葉子的節(jié)點 tree 及其相關(guān)取值函數(shù)勺拣。

有序 list 構(gòu)造方法
(define (adjoin-set x set)
  ;如果 set 為空奶赠,則返回以 x 作為唯一元素的 list
  (cond ((null? set) 
      (list x)) 
    ;如果 set 的第一個元素的 weight 大于 x 的 weight,則將 x 和 set 組合成一個新的 list 返回
    ((> (weight (car set)) (weight x))
      (cons x set)) 
    ; 否則將 set 的以第一個只取出药有,讓后遞歸調(diào)用 `adjoin-set`
    (else (cons (car set) (adjoin-set x (cdr set)))))) 
(define (make-leaf-set pairs)
  (if (null? pairs)
    '()
    (let ((pair (car pairs)))
      (adjoin-set (make-leaf (car pair) (cadr pair))
        (make-leaf-set (cdr pairs))))))

adjoin-set 的功能就是 x 插入到有序 list set 中车柠,保證插入后的 list 仍然有序。lisp 中的 cond 可理解為 其他語言中的 switch塑猖,而 cons 可理解為將兩個元素結(jié)合成一個 list。 乍一看這個所謂“插入”元素的方法有點奇怪谈跛,而且沒有用任何臨時變量羊苟。其思路將整個插入的過程用遞歸調(diào)用的方式表示: 用過程(函數(shù))代替了臨時變量。舉了例子:(adjoin-set 3 '(1 2))感憾,執(zhí)行順序是:

(cons 1 (adjoin-set 3 '(2)))
(cons 1 (cons 2 (adjoin-set 3 '())))
(cons 1 (cons 2 (cons 3 '())))
(cons 1 (cons 2 '(3)))
(cons 1 '(2 3))
'(1 2 3)

可以看到在執(zhí)行序列中蜡励,推遲 cons 的執(zhí)行,用參數(shù)求值壓棧從而省去了臨時變量阻桅。在 make-leaf-set 中思路也一樣:不斷地從 paris 中取元素凉倚,交給 adjoin-set 插入到 list 中。整個編寫過程中基本上用程序流暢地表達(dá)了我們的解題思路嫂沉。

Huffman樹構(gòu)造方法

在插入元素這種簡單的問題中函數(shù)式威力還遠(yuǎn)遠(yuǎn)沒有體現(xiàn)出來稽寒,請看下面構(gòu)造 Huffman樹 的函數(shù)實現(xiàn):

(define (make-tree leaves)
  (cond ((or (null? (car leaves)) (null? (cadr leaves)))
      (error "leaves is not enough"))
    ((null? (cddr leaves))
      (make-code-tree (car leaves) (cadr leaves)))
    (else (make-tree (adjoin-set (make-code-tree (car leaves) (cadr leaves)) (cddr leaves))))))

幾行代碼就將構(gòu)造 Huffman樹 的核心邏輯表達(dá)清楚了:將按 weight 升序 leaves 的前兩個拿出來做成一個 tree node,adjoin-set 到剩下的 leaves 中趟章,然后不斷重復(fù)這個操作杏糙,直到 leaves 中只剩下兩個元素,將這兩個元素最為 最終 Huffman樹 的左右子樹蚓土,然后返回宏侍。怎么樣?一氣呵成蜀漆。

編碼

對 Huffman樹 遍歷編碼的實現(xiàn)也是精煉得有種思維的美感:
先進行左子樹遍歷谅河,直到找到葉子節(jié)點,構(gòu)造成結(jié)果 list 中一個元素确丢,然后回到上一層遞歸绷耍,進入右子樹,不斷重復(fù)直到遍歷完所有節(jié)點蠕嫁。

(define (encode tree)
  (define (visit n bits)
    (if (leaf? n)
      ; 找到了一個葉子節(jié)點
      (cons (symbol-leaf n) bits)
      ; 用 cons 對 visit 的遞歸調(diào)用
      (cons (visit (left-branch n) (cons 0 bits))
        (visit (right-branch n) (cons 1 bits)))))
  (visit tree '()))

;測試
(define leaf-set (make-leaf-set '((A 1) (B 3) (C 2))))
(define tree (make-tree leaf-set))
(encode tree) ; outputs: ((b 0) (a 0 1) c 1 1)

詳細(xì)代碼請進 github

ps: 本篇用到部分《計算機程序的構(gòu)造與解析》代碼锨天。強烈建議大家學(xué)習(xí) MIT 的這門公開課。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末剃毒,一起剝皮案震驚了整個濱河市病袄,隨后出現(xiàn)的幾起案子搂赋,更是在濱河造成了極大的恐慌,老刑警劉巖益缠,帶你破解...
    沈念sama閱讀 219,490評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件脑奠,死亡現(xiàn)場離奇詭異,居然都是意外死亡幅慌,警方通過查閱死者的電腦和手機宋欺,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來胰伍,“玉大人齿诞,你說我怎么就攤上這事÷钭猓” “怎么了祷杈?”我有些...
    開封第一講書人閱讀 165,830評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長渗饮。 經(jīng)常有香客問我但汞,道長,這世上最難降的妖魔是什么互站? 我笑而不...
    開封第一講書人閱讀 58,957評論 1 295
  • 正文 為了忘掉前任私蕾,我火速辦了婚禮,結(jié)果婚禮上胡桃,老公的妹妹穿的比我還像新娘踩叭。我一直安慰自己,他們只是感情好标捺,可當(dāng)我...
    茶點故事閱讀 67,974評論 6 393
  • 文/花漫 我一把揭開白布懊纳。 她就那樣靜靜地躺著,像睡著了一般亡容。 火紅的嫁衣襯著肌膚如雪嗤疯。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,754評論 1 307
  • 那天闺兢,我揣著相機與錄音茂缚,去河邊找鬼。 笑死屋谭,一個胖子當(dāng)著我的面吹牛脚囊,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播桐磁,決...
    沈念sama閱讀 40,464評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼悔耘,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了我擂?” 一聲冷哼從身側(cè)響起衬以,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤缓艳,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后看峻,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體阶淘,經(jīng)...
    沈念sama閱讀 45,847評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,995評論 3 338
  • 正文 我和宋清朗相戀三年互妓,在試婚紗的時候發(fā)現(xiàn)自己被綠了溪窒。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,137評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡冯勉,死狀恐怖澈蚌,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情灼狰,我是刑警寧澤惜浅,帶...
    沈念sama閱讀 35,819評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站伏嗜,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏伐厌。R本人自食惡果不足惜承绸,卻給世界環(huán)境...
    茶點故事閱讀 41,482評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望挣轨。 院中可真熱鬧军熏,春花似錦、人聲如沸卷扮。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,023評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽晤锹。三九已至摩幔,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間鞭铆,已是汗流浹背或衡。 一陣腳步聲響...
    開封第一講書人閱讀 33,149評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留车遂,地道東北人封断。 一個月前我還...
    沈念sama閱讀 48,409評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像舶担,于是被迫代替她去往敵國和親坡疼。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,086評論 2 355

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

  • 3.4 說說相等和內(nèi)部表示 在Lisp中主要有5種相等斷言衣陶,因為不是所有的對象被創(chuàng)建的時候都是相等意義上的相等柄瑰。數(shù)...
    geoeee閱讀 1,819評論 0 6
  • Lisp的本質(zhì) - climbdream的個人空間 - 開源中國社區(qū)https://my.oschina.net/...
    葡萄喃喃囈語閱讀 700評論 0 10
  • 說明 函數(shù)式編程和面向?qū)ο缶幊炭梢哉f是編程的兩大宗教闸氮,猶如編輯器之爭一樣,之間口角不斷狱意。我雖然靠著OOP的主力語言...
    lingyv閱讀 1,690評論 1 14
  • 第一部分Common Lisp介紹第1章 介紹一下Lisp你在學(xué)的時候覺得已經(jīng)明白了湖苞,寫的時候更加確信了解了,教別...
    geoeee閱讀 2,947評論 5 8
  • 1.不過做什么事详囤,尤其是一個重大的決定财骨,一定要多問幾個為什么。最好把它寫下來藏姐,放到一個比較明顯的地方隆箩,時刻提醒自己...
    弘毅浪跡天涯閱讀 69評論 0 0