遞歸:簡潔高效的編程技巧

1. 如何理解遞歸并巍?

遞歸是一種應用非常廣泛的編程技巧,比如 DFS 深度優(yōu)先搜索刽射、前中后序二叉樹遍歷等都是使用遞歸剃执。

一個遞歸求解問題的分解過程,去的過程叫“遞”肾档,回來的過程叫“歸”∷状龋基本上,所有的遞歸問題都可以用遞推公式來表示低千。比如:

f(n) = f(n-1) + 1馏颂,其中 f(1) = 1。

遞歸的優(yōu)點和缺點:

  • 優(yōu)點:遞歸代碼的表達力很強难审,寫起來非常簡潔亿絮。
  • 缺點:空間復雜度高、有堆棧溢出的風險黔姜、存在重復計算蒂萎、過多的函數(shù)調用會耗時較多等問題。

2. 遞歸需要滿足的三個條件

  1. 一個問題的解可以分解為幾個子問題的解

  2. 這個問題與分解之后的子問題五慈,除了數(shù)據(jù)規(guī)模不同泻拦,求解思路完全一樣

  3. 存在遞歸終止條件

3. 如何編寫遞歸代碼?

寫遞歸代碼最關鍵的是寫出遞推公式争拐,找到終止條件陆错,剩下將遞推公式轉化為代碼就容易了。

假如這里有 n 個臺階音瓷,每次你可以跨 1 個臺階或者 2 個臺階夹抗,請問走這 n 個臺階有多少種走法?

仔細想下杏愤,n 個臺階的走法就等于先走 1 階后,n-1 個臺階的走法通殃,加上先走 2 階后厕宗,n-2 個臺階的走法。遞歸終止條件是 f(1)=1曲聂,f(2)=2佑惠。其實這就是著名的斐波那契數(shù)列。

f(n) = f(n-1) + f(n-2)旭咽,其中 f(1) = 1赌厅,f(2) = 2。

最終的遞歸代碼:

int f(int n) {
    if (n == 1) {
        return 1;
    }
    if (n == 2) {
        return 2;
    }
    return f(n-1) + f(n-2);
}

總結一下请垛,寫遞歸代碼的關鍵就是找到如何將大問題分解為小問題的規(guī)律洽议,并且基于此寫出遞推公式,然后再推敲終止條件混稽,最后將遞推公式和終止條件翻譯成代碼审胚。

只要遇到遞歸,就把它抽象成一個遞推公式膳叨,不用想一層層的調用關系菲嘴,不要試圖用人腦去分解遞歸的每個步驟汰翠。

4. 警惕堆棧溢出和重復計算

函數(shù)調用會使用棧來保存臨時變量昭雌。每調用一個函數(shù),都會將臨時變量封裝為棧幀壓入內存棧佛纫,等函數(shù)執(zhí)行完成返回時总放,才出棧。如果遞歸求解的數(shù)據(jù)規(guī)模很大攒盈,調用層次很深哎榴,一直壓入棧,就會有堆棧溢出的風險迎变。

可以通過限制遞歸調用的最大深度的方式飘言,來解決堆棧溢出。

比如上面的遞歸谆吴,想要計算 f(5)苛预,需要先計算 f(4) 和 f(3),而計算 f(4) 還需要計算 f(3)腻菇,因此昔馋,f(3) 就被計算了很多次,這就是重復計算問題丘薛。

可以通過一個數(shù)據(jù)結構(比如散列表)來保存已經求解過的值垄提,解決重復計算周拐。

5. 將遞歸代碼改為非遞歸

籠統(tǒng)地講凰兑,如果我們自己在內存堆上實現(xiàn)棧审丘,手動模擬入棧滩报、出棧過程,這樣任何遞歸代碼都可以改寫成看上去不是遞歸代碼的樣子脓钾。

將斐波那契數(shù)列改為非遞歸方式實現(xiàn):

int f(int n) {
    if (n == 1) {
        return 1;
    }
  if (n == 2) {
    return 2;
  }
  int ret = 0;
  int pre = 2;
  int prepre = 1;
  for (int i = 3; i < n; i++) {
    ret = pre + prepre;
    prepre = pre;
    pre = ret;
  }
  return ret;
}

練習題

  • 編程實現(xiàn)求階乘n!
  • 編程實現(xiàn)一組數(shù)據(jù)集合的全排列
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末可训,一起剝皮案震驚了整個濱河市握截,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌谨胞,老刑警劉巖胯努,帶你破解...
    沈念sama閱讀 211,265評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異蒲讯,居然都是意外死亡恬汁,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,078評論 2 385
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來约巷,“玉大人,你說我怎么就攤上這事踩麦。” “怎么了贫橙?”我有些...
    開封第一講書人閱讀 156,852評論 0 347
  • 文/不壞的土叔 我叫張陵卢肃,是天一觀的道長才顿。 經常有香客問我,道長幅垮,這世上最難降的妖魔是什么尾组? 我笑而不...
    開封第一講書人閱讀 56,408評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮匕争,結果婚禮上爷耀,老公的妹妹穿的比我還像新娘。我一直安慰自己跑杭,他們只是感情好咆耿,可當我...
    茶點故事閱讀 65,445評論 5 384
  • 文/花漫 我一把揭開白布萨螺。 她就那樣靜靜地躺著,像睡著了一般慰技。 火紅的嫁衣襯著肌膚如雪吻商。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,772評論 1 290
  • 那天乌叶,我揣著相機與錄音盆偿,去河邊找鬼。 笑死准浴,一個胖子當著我的面吹牛事扭,可吹牛的內容都是我干的。 我是一名探鬼主播兄裂,決...
    沈念sama閱讀 38,921評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼句旱,長吁一口氣:“原來是場噩夢啊……” “哼阳藻!你這毒婦竟也來了晰奖?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 37,688評論 0 266
  • 序言:老撾萬榮一對情侶失蹤腥泥,失蹤者是張志新(化名)和其女友劉穎匾南,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蛔外,經...
    沈念sama閱讀 44,130評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,467評論 2 325
  • 正文 我和宋清朗相戀三年夹厌,在試婚紗的時候發(fā)現(xiàn)自己被綠了豹爹。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,617評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡矛纹,死狀恐怖臂聋,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情或南,我是刑警寧澤孩等,帶...
    沈念sama閱讀 34,276評論 4 329
  • 正文 年R本政府宣布,位于F島的核電站采够,受9級特大地震影響肄方,放射性物質發(fā)生泄漏。R本人自食惡果不足惜蹬癌,卻給世界環(huán)境...
    茶點故事閱讀 39,882評論 3 312
  • 文/蒙蒙 一权她、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧逝薪,春花似錦隅要、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,740評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至感局,卻和暖如春尼啡,著一層夾襖步出監(jiān)牢的瞬間暂衡,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,967評論 1 265
  • 我被黑心中介騙來泰國打工崖瞭, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留狂巢,地道東北人。 一個月前我還...
    沈念sama閱讀 46,315評論 2 360
  • 正文 我出身青樓书聚,卻偏偏與公主長得像唧领,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子雌续,可洞房花燭夜當晚...
    茶點故事閱讀 43,486評論 2 348

推薦閱讀更多精彩內容

  • 遞歸:如何用三行代碼找到“最終推薦人” 推薦注冊返傭金的這個功能我想你應該不陌生吧斩个?現(xiàn)在很多 App 都有這個功能...
    GhostintheCode閱讀 1,243評論 0 6
  • 遞歸思想 遞歸指的是一個過程:函數(shù)不斷引用自身,直到引用的對象已知驯杜。遞歸就是在函數(shù)內部調用自己的函數(shù)被稱之為遞歸受啥。...
    Artifacts閱讀 317評論 0 0
  • 在學習「數(shù)據(jù)結構和算法」的過程中,因為人習慣了平鋪直敘的思維方式鸽心,所以「遞歸」與「動態(tài)規(guī)劃」這種帶循環(huán)概念(繞來繞...
    五分鐘學算法閱讀 1,908評論 1 34
  • 兒子雖然有時惹我們生氣滚局,但是生活的點點滴滴里露出了兒子在長大,生活中的他在不斷進步與成長顽频!在生活方面兒子...
    only王梓曄閱讀 176評論 0 1
  • 2017.10.6藤肢。無為。焦點解決第69天糯景。焦慮產生的根本原因嘁圈,在于大家沒有找到一條與自己內心深度溝通的渠道,沒有...
    無為wyw閱讀 213評論 0 1