日日操夜夜添-日日操影院-日日草夜夜操-日日干干-精品一区二区三区波多野结衣-精品一区二区三区高清免费不卡

公告:魔扣目錄網(wǎng)為廣大站長提供免費(fèi)收錄網(wǎng)站服務(wù),提交前請做好本站友鏈:【 網(wǎng)站目錄:http://www.ylptlb.cn 】, 免友鏈快審服務(wù)(50元/站),

點(diǎn)擊這里在線咨詢客服
新站提交
  • 網(wǎng)站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會(huì)員:747

 

一、什么是遞歸?

自己調(diào)用自己,當(dāng)業(yè)務(wù)邏輯符合以下三個(gè)條件的時(shí)候,就可以考慮使用遞歸來實(shí)現(xiàn)。

  • 一個(gè)問題可以分解為多個(gè)子問題;
  • 當(dāng)前問題與其子問題除了數(shù)據(jù)規(guī)模不同外,求解思路完全一樣;
  • 存在遞歸終止條件;

二、為什么要使用遞歸?

理論上,任何遞歸代碼都可以使用非遞歸的方式進(jìn)行實(shí)現(xiàn),那么為什么還要使用遞歸呢?

遞歸求解本質(zhì)上是使用系統(tǒng)提供的棧來實(shí)現(xiàn)的,由系統(tǒng)為我們自動(dòng)遞歸求解;如果改成非遞歸的方式,那么我們其實(shí)就是在手動(dòng)遞歸,本質(zhì)上這兩種方式?jīng)]有任何區(qū)別。但是相同的邏輯,兩份代碼,遞歸實(shí)現(xiàn)明顯更加地簡潔,更加有利系統(tǒng)的維護(hù)。

三、如何使用遞歸?

我們以一個(gè)實(shí)際的例子來演示如何使用遞歸。

假設(shè),有n個(gè)臺(tái)階,我們一步可以跨一個(gè)臺(tái)階,也可以一步跨兩個(gè)臺(tái)階,那么這n個(gè)臺(tái)階總共有多少種跨法?

我們按照遞歸使用的三個(gè)條件來分析下:

  • 是否可以分解為子問題?
  • 是的,我們跨上第n階臺(tái)階時(shí),要么是一步一個(gè)跨上去的,要么就是一步兩個(gè)跨上去的,所以問題就分解為“跨上n-1個(gè)臺(tái)階有多少種跨法”+“跨上n-2個(gè)臺(tái)階有多少種跨法”,遞推公式可以表示為f(n)=f(n-1)+f(n-2);
  • 當(dāng)前問題和子問題的求解思路是否一樣?
  • 是的
  • 是否存在終止條件?
  • 是的,當(dāng)n為1的時(shí)候,就一種跨法,當(dāng)n為2的時(shí)候有兩種跨法,當(dāng)n為3或者4的時(shí)候,可以拆解為子問題,所以終止條件就是f(1)=1,f(2)=2;

分析完之后,我們就得到了帶終止條件的遞推公式,再轉(zhuǎn)換為代碼即可:

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

遞歸的求解過程確實(shí)很難一下子就理解清楚,這是由于我們?nèi)四X的機(jī)制造成的,誰都會(huì)感覺這樣。

我們應(yīng)該假設(shè)子問題f(n-1)和f(n-2)已經(jīng)得到求解,然后在此基礎(chǔ)上去求解當(dāng)前的問題f(n),得到一個(gè)遞推公式,如此只思考兩層之間的關(guān)聯(lián)關(guān)系會(huì)簡單很多,千萬不要試圖去分解和理解每個(gè)遞歸層次的過程。

四、使用遞歸需要注意什么?

4.1 堆棧溢出

當(dāng)遞歸調(diào)用深度過深的時(shí)候,就很容易出現(xiàn)棧溢出的問題,此時(shí)最好的方法就是根據(jù)業(yè)務(wù)場景邏輯和JVM的棧大小確定一個(gè)合理的最大遞歸深度,當(dāng)超過該深度值的時(shí)候,程序拋出異常;

4.2 重復(fù)計(jì)算

子問題的子問題可能會(huì)出現(xiàn)重復(fù)的問題,比如f(6)=f(5)+f(4),其中,f(5)和f(4)都會(huì)有子問題f(3),那么f(3)應(yīng)該是不需要重復(fù)計(jì)算的。

我們可以將求解過的問題結(jié)果保存到散列表中,當(dāng)遞歸調(diào)用執(zhí)行時(shí),先檢查該問題是否已經(jīng)求解過了,如果是那么直接從散列表中獲取結(jié)果返回即可,只有沒有求解過的問題,才繼續(xù)遞歸調(diào)用;

int countSteps(int n){
    if(n == 1) {
        return 1;
    }
    if(n == 2) {
        return 2;
    }
    
    // 先從散列表中檢查是否已經(jīng)對(duì)f(n)求解過了
    if(resultMap.get(n) != null){
        return resultMap.get(n);
    }
    int result = f(n-1) + f(n-2);
    // 將當(dāng)前問題f(n)結(jié)果保存到散列表
    resultMap.put(n,result);
    return result;
}

4.3 時(shí)空復(fù)雜度

空間復(fù)雜度方面,因?yàn)檫f歸深度決定著棧的使用程度,所以空間復(fù)雜度為O(n);

時(shí)間復(fù)雜度上面,雖然入棧和出棧都是O(1),但是如果遞歸深度較大的話,函數(shù)上下文切換等造成的時(shí)間開銷也會(huì)比較可觀;

4.4 遞歸環(huán)

有時(shí)候,可能由于臟數(shù)據(jù)的問題,會(huì)導(dǎo)致遞歸環(huán)的存在,比如在找原始推薦人的時(shí)候,A推薦了B,B推薦了C,C推薦了D,然后有一條臟數(shù)據(jù),導(dǎo)致A的推薦人是D,那么在遞歸尋找原始推薦人的時(shí)候,就會(huì)陷入環(huán)中,可以設(shè)置一個(gè)最大遞歸深度來解決這個(gè)問題。

分享到:
標(biāo)簽:遞歸 算法
用戶無頭像

網(wǎng)友整理

注冊時(shí)間:

網(wǎng)站:5 個(gè)   小程序:0 個(gè)  文章:12 篇

  • 51998

    網(wǎng)站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會(huì)員

趕快注冊賬號(hào),推廣您的網(wǎng)站吧!
最新入駐小程序

數(shù)獨(dú)大挑戰(zhàn)2018-06-03

數(shù)獨(dú)一種數(shù)學(xué)游戲,玩家需要根據(jù)9

答題星2018-06-03

您可以通過答題星輕松地創(chuàng)建試卷

全階人生考試2018-06-03

各種考試題,題庫,初中,高中,大學(xué)四六

運(yùn)動(dòng)步數(shù)有氧達(dá)人2018-06-03

記錄運(yùn)動(dòng)步數(shù),積累氧氣值。還可偷

每日養(yǎng)生app2018-06-03

每日養(yǎng)生,天天健康

體育訓(xùn)練成績評(píng)定2018-06-03

通用課目體育訓(xùn)練成績評(píng)定