Elixir 簡明筆記(十七) --- 控制結(jié)構(gòu)之遞歸與迭代

計算機顧名思義,執(zhí)行計算功能彬呻。然而通常計算機是通過重復計算得出結(jié)果衣陶。例如計算從1到100的求和柄瑰,就是一個個數(shù)進行累加。這樣的重復性工作剪况,人類顯然不太擅長教沾,所以人發(fā)明了算法。高斯小朋友就實現(xiàn)了一個等差數(shù)列的求和算法译断。

既然計算機通過快速的重復計算得出結(jié)果详囤。那么對數(shù)據(jù)源進行循環(huán)處理就是一種基本的控制結(jié)構(gòu)。經(jīng)典的循環(huán)控制都是通過while镐作,for語句實現(xiàn)藏姐。很可惜,elixr中并沒有這樣的語句该贾。不過不用擔心羔杨,elixir提供了優(yōu)雅而強大的遞歸來進行循環(huán)迭代處理。

遞歸

在函數(shù)內(nèi)調(diào)用函數(shù)自身即為遞歸杨蛋。正常情況下兜材,打印一些一些自然數(shù)可以這么寫:

# python
In [1]: for i in range(1, 6):
   ...:     print i
   ...:
1
2
3
4
5

In [4]: n = 1

In [5]: while n <= 5:
   ...:     print n
   ...:     n += 1
   ...:
1
2
3
4
5

Elixir 中使用遞歸則為:

defmodule Nums do

    def print(1), do: IO.puts 1

    def print(n) do
     print(n - 1)
     IO.puts n
    end

end

iex(1)> Nums.print(5)
1
2
3
4
5
:ok

列表遞歸

Lisp擁有很高的聲譽,它本身就是指列表進行計算逞力。前面我們提及曙寡,List的實現(xiàn)是鏈表結(jié)構(gòu),因此讀取列表第一個元素將會很快寇荧。elixir還提供了|操作符举庶。下面就使用列表來完成加和計算:

defmodule Math do
    
    def sum([]) do
        0
    end 

    def sum([head|tail]) do
        head + sum(tail)
    end

end


IO.puts Math.sum([1, 2, 3, 4, 5])

尾遞歸

使用遞歸來做循環(huán)控制是函數(shù)式語言的一大特點。別的語言通常在處理遞歸的時候存在著一個最大遞歸深度揩抡。而elixir對遞歸在編譯上做了優(yōu)化户侥。當然并不是針對所有遞歸,而是專門指尾遞歸(tail recursion)峦嗤。至于尾遞歸和遞歸的差別蕊唐,可以閱讀遞歸和尾遞歸。其形式大概就是自己調(diào)用自己的時候烁设,并不是作為表達式替梨,大概形式如下:

def original_fun(...) do
    ...
    another_fun(...)    # Tail call 
end

上述加和操作用尾遞歸重寫如下:

defmodule Math do
    
    def sum(list) do
        _sum(list, 0)
    end

    defp _sum([], accumulator) do
        accumulator
    end

    defp _sum([head|tail], accumulator) do
        _sum(tail, head + accumulator)
    end

end


IO.puts Math.sum([1, 2, 3, 4, 5])

更多遞歸

上述遞歸的控制基于函數(shù)參數(shù)的模式匹配,關于條件的控制装黑,還有case和cond宏副瀑,下面結(jié)合這些控制結(jié)構(gòu)來重寫上述的求和計算。

Lisp中常用cond結(jié)構(gòu)曹体,下面情況cond的結(jié)果:

defmodule Math do
    
    def sum(list) do
        _sum(list, 0)
    end

    def _sum(list, accumulator) do
        
        cond do
            list == [] -> 
                accumulator
            
            [head|tail] = list -> 
                _sum(tail, head+accumulator)
        end

    end

end


IO.puts Math.sum([1, 2, 3, 4, 5])

再看case的結(jié)果:

defmodule Math do
    
    def sum(list) do
        _sum(list, 0)
    end

    def _sum(list, accumulator) do
        
        case list do
            [] -> accumulator
            [head|tail] -> _sum(tail, head + accumulator)
        end
    end
end


IO.puts Math.sum([1, 2, 3, 4, 5])

由此可見俗扇,使用遞歸實現(xiàn)循環(huán)控制,遇到條件判斷的時候箕别,即可以使用函數(shù)參數(shù)的模式匹配铜幽,也可以使用cond和case方式進行判斷滞谢。其中cond和case內(nèi)部邏輯分支也是基于模式匹配。

列表解析

todo

總而言之除抛,使用elixir的遞歸實現(xiàn)迭代功能狮杨,盡量使用尾遞歸。尾遞歸經(jīng)過編譯器優(yōu)化到忽,不會因為調(diào)用棧出現(xiàn)問題橄教,也不會引起額外的內(nèi)存開銷。普通遞歸則會出現(xiàn)最大遞歸棧的問題喘漏。實際上护蝶,函數(shù)式語言的特點就是函數(shù)。在實現(xiàn)循環(huán)的功能翩迈,Elixir提供了一些高階函數(shù)用來處理持灰,通過這些高階函數(shù)的抽象,隱藏了遞歸調(diào)用的細節(jié)负饲。開發(fā)者掌握這些高階函數(shù)堤魁,更有利于實現(xiàn)功能需求。Enum和Stream提供了大部分所需要的函數(shù)返十,下一節(jié)繼續(xù)探索高階函數(shù)妥泉。

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市洞坑,隨后出現(xiàn)的幾起案子盲链,更是在濱河造成了極大的恐慌,老刑警劉巖检诗,帶你破解...
    沈念sama閱讀 221,820評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件匈仗,死亡現(xiàn)場離奇詭異瓢剿,居然都是意外死亡逢慌,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,648評論 3 399
  • 文/潘曉璐 我一進店門间狂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來攻泼,“玉大人,你說我怎么就攤上這事鉴象∶Σぃ” “怎么了?”我有些...
    開封第一講書人閱讀 168,324評論 0 360
  • 文/不壞的土叔 我叫張陵纺弊,是天一觀的道長牛欢。 經(jīng)常有香客問我,道長淆游,這世上最難降的妖魔是什么傍睹? 我笑而不...
    開封第一講書人閱讀 59,714評論 1 297
  • 正文 為了忘掉前任隔盛,我火速辦了婚禮,結(jié)果婚禮上拾稳,老公的妹妹穿的比我還像新娘吮炕。我一直安慰自己,他們只是感情好访得,可當我...
    茶點故事閱讀 68,724評論 6 397
  • 文/花漫 我一把揭開白布龙亲。 她就那樣靜靜地躺著,像睡著了一般悍抑。 火紅的嫁衣襯著肌膚如雪鳄炉。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,328評論 1 310
  • 那天搜骡,我揣著相機與錄音迎膜,去河邊找鬼。 笑死浆兰,一個胖子當著我的面吹牛磕仅,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播簸呈,決...
    沈念sama閱讀 40,897評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼榕订,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了蜕便?” 一聲冷哼從身側(cè)響起劫恒,我...
    開封第一講書人閱讀 39,804評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎轿腺,沒想到半個月后两嘴,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,345評論 1 318
  • 正文 獨居荒郊野嶺守林人離奇死亡族壳,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,431評論 3 340
  • 正文 我和宋清朗相戀三年憔辫,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片仿荆。...
    茶點故事閱讀 40,561評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡贰您,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出拢操,到底是詐尸還是另有隱情锦亦,我是刑警寧澤,帶...
    沈念sama閱讀 36,238評論 5 350
  • 正文 年R本政府宣布令境,位于F島的核電站杠园,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏舔庶。R本人自食惡果不足惜抛蚁,卻給世界環(huán)境...
    茶點故事閱讀 41,928評論 3 334
  • 文/蒙蒙 一玲昧、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧篮绿,春花似錦孵延、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,417評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至吼虎,卻和暖如春犬钢,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背思灰。 一陣腳步聲響...
    開封第一講書人閱讀 33,528評論 1 272
  • 我被黑心中介騙來泰國打工玷犹, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人洒疚。 一個月前我還...
    沈念sama閱讀 48,983評論 3 376
  • 正文 我出身青樓歹颓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親油湖。 傳聞我的和親對象是個殘疾皇子巍扛,可洞房花燭夜當晚...
    茶點故事閱讀 45,573評論 2 359

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