遞歸和動態(tài)規(guī)劃

暴力遞歸
1.把問題轉(zhuǎn)化為規(guī)模最小的同類問題的子問題
2.有明確的不需要繼續(xù)進(jìn)行遞歸的條件(base,case)
3.有當(dāng)?shù)玫阶訂栴}的結(jié)果之后的決策問題
4.不記錄每一個(gè)子問題的解

動態(tài)規(guī)劃
1.從暴力遞歸中來
2.將每一個(gè)子問題的解記錄下來颈走,避免重復(fù)計(jì)算
3.把暴力遞歸的過程阐污,抽象成狀態(tài)表達(dá)
4.并且存在化簡狀態(tài)表達(dá)芽卿,使其更加簡潔

背包問題:

先按照題意暴力遞歸沸呐,然后去除題意,改為dp.

給定兩個(gè)數(shù)組w和v挪钓, 兩個(gè)數(shù)組長度相等栅屏, w[i]表示第i件商品的
重量馆里, v[i]表示第i件商品的價(jià)值。
再給定一個(gè)整數(shù)bag筒扒, 要求你挑選商品的重量加起來一定不能超
過bag怯邪, 返回滿足這個(gè)條件下, 你能獲得的最大價(jià)值.


package basic_class_07;

public class Code_09_Knapsack {

    public static int maxValue1(int[] c, int[] p, int bag) {
        return process1(c, p, 0, 0, bag);
    }

    public static int process1(int[] c, int[] p, int i, int cost, int bag) {
        if (cost > bag) {
            return Integer.MIN_VALUE;
        }
        if (i == c.length) {
            return 0;
        }
        return Math.max(process1(c, p, i + 1, cost, bag), p[i] + process1(c, p, i + 1, cost + c[i], bag));
    }

    public static int maxValue2(int[] c, int[] p, int bag) {
        int[][] dp = new int[c.length + 1][bag + 1];
        for (int i = c.length - 1; i >= 0; i--) {
            for (int j = bag; j >= 0; j--) {
                dp[i][j] = dp[i + 1][j];
                if (j + c[i] <= bag) {
                    dp[i][j] = Math.max(dp[i][j], p[i] + dp[i + 1][j + c[i]]);
                }
            }
        }
        return dp[0][0];
    }

    public static void main(String[] args) {
        int[] c = { 3, 2, 4, 7 };
        int[] p = { 5, 6, 3, 19 };
        int bag = 11;
        System.out.println(maxValue1(c, p, bag));
        System.out.println(maxValue2(c, p, bag));
    }

}

題目:數(shù)組中選和問題

下面代碼中花墩,一種是暴力遞歸悬秉,一種是動態(tài)規(guī)劃

給你一個(gè)數(shù)組arr澄步, 和一個(gè)整數(shù)aim。 如果可以任意選擇arr中的
數(shù)字和泌, 能不能累加得到aim村缸, 返回true或者false



public class Code_08_Money_Problem {

    public static boolean money1(int[] arr, int aim) {
        return process1(arr, 0, 0, aim);
    }

    public static boolean process1(int[] arr, int i, int sum, int aim) {
        if (sum == aim) {
            return true;
        }
        if (i == arr.length) {
            return false;
        }
        return process1(arr, i + 1, sum, aim) || process1(arr, i + 1, sum + arr[i], aim);
    }

    public static boolean money2(int[] arr, int aim) {
        boolean[][] dp = new boolean[arr.length + 1][aim + 1];
        for (int i = 0; i < dp.length; i++) {
            dp[i][aim] = true;
        }
        for (int i = arr.length - 1; i >= 0; i--) {
            for (int j = aim - 1; j >= 0; j--) {
                dp[i][j] = dp[i + 1][j];
                if (j + arr[i] <= aim) {
                    dp[i][j] = dp[i][j] || dp[i + 1][j + arr[i]];
                }
            }
        }
        return dp[0][0];
    }

    public static void main(String[] args) {
        int[] arr = { 1, 4, 8 };
        int aim = 12;
        System.out.println(money1(arr, aim));
        System.out.println(money2(arr, aim));
    }

}

最小路徑和問題

下面代碼中一種是暴力遞歸,一種是動態(tài)規(guī)劃

給你一個(gè)二維數(shù)組武氓, 二維數(shù)組中的每個(gè)數(shù)都是正數(shù)梯皿, 要求從左上
角走到右下角, 每一步只能向右或者向下县恕。 沿途經(jīng)過的數(shù)字要累
加起來东羹。 返回最小的路徑和。



public class Code_07_MinPath {

    public static int minPath1(int[][] matrix) {
        return process1(matrix, matrix.length - 1, matrix[0].length - 1);
    }

    public static int process1(int[][] matrix, int i, int j) {
        int res = matrix[i][j];
        if (i == 0 && j == 0) {
            return res;
        }
        if (i == 0 && j != 0) {
            return res + process1(matrix, i, j - 1);
        }
        if (i != 0 && j == 0) {
            return res + process1(matrix, i - 1, j);
        }
        return res + Math.min(process1(matrix, i, j - 1), process1(matrix, i - 1, j));
    }

    public static int minPath2(int[][] m) {
        if (m == null || m.length == 0 || m[0] == null || m[0].length == 0) {
            return 0;
        }
        int row = m.length;
        int col = m[0].length;
        int[][] dp = new int[row][col];
        dp[0][0] = m[0][0];
        for (int i = 1; i < row; i++) {
            dp[i][0] = dp[i - 1][0] + m[i][0];
        }
        for (int j = 1; j < col; j++) {
            dp[0][j] = dp[0][j - 1] + m[0][j];
        }
        for (int i = 1; i < row; i++) {
            for (int j = 1; j < col; j++) {
                dp[i][j] = Math.min(dp[i - 1][j], dp[i][j - 1]) + m[i][j];
            }
        }
        return dp[row - 1][col - 1];
    }

    // for test
    public static int[][] generateRandomMatrix(int rowSize, int colSize) {
        if (rowSize < 0 || colSize < 0) {
            return null;
        }
        int[][] result = new int[rowSize][colSize];
        for (int i = 0; i != result.length; i++) {
            for (int j = 0; j != result[0].length; j++) {
                result[i][j] = (int) (Math.random() * 10);
            }
        }
        return result;
    }

    public static void main(String[] args) {
        int[][] m = { { 1, 3, 5, 9 }, { 8, 1, 3, 4 }, { 5, 0, 6, 1 }, { 8, 8, 4, 0 } };
        System.out.println(minPath1(m));
        System.out.println(minPath2(m));

        m = generateRandomMatrix(6, 7);
        System.out.println(minPath1(m));
        System.out.println(minPath2(m));
    }
}

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末忠烛,一起剝皮案震驚了整個(gè)濱河市属提,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌美尸,老刑警劉巖冤议,帶你破解...
    沈念sama閱讀 211,948評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異火惊,居然都是意外死亡求类,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,371評論 3 385
  • 文/潘曉璐 我一進(jìn)店門屹耐,熙熙樓的掌柜王于貴愁眉苦臉地迎上來尸疆,“玉大人,你說我怎么就攤上這事惶岭∈偃酰” “怎么了?”我有些...
    開封第一講書人閱讀 157,490評論 0 348
  • 文/不壞的土叔 我叫張陵按灶,是天一觀的道長症革。 經(jīng)常有香客問我,道長鸯旁,這世上最難降的妖魔是什么噪矛? 我笑而不...
    開封第一講書人閱讀 56,521評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮铺罢,結(jié)果婚禮上艇挨,老公的妹妹穿的比我還像新娘。我一直安慰自己韭赘,他們只是感情好缩滨,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,627評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般脉漏。 火紅的嫁衣襯著肌膚如雪苞冯。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,842評論 1 290
  • 那天侧巨,我揣著相機(jī)與錄音舅锄,去河邊找鬼。 笑死司忱,一個(gè)胖子當(dāng)著我的面吹牛巧娱,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播烘贴,決...
    沈念sama閱讀 38,997評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼禁添,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了桨踪?” 一聲冷哼從身側(cè)響起老翘,我...
    開封第一講書人閱讀 37,741評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎锻离,沒想到半個(gè)月后铺峭,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,203評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡汽纠,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,534評論 2 327
  • 正文 我和宋清朗相戀三年卫键,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片虱朵。...
    茶點(diǎn)故事閱讀 38,673評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡莉炉,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出碴犬,到底是詐尸還是另有隱情絮宁,我是刑警寧澤,帶...
    沈念sama閱讀 34,339評論 4 330
  • 正文 年R本政府宣布服协,位于F島的核電站绍昂,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏偿荷。R本人自食惡果不足惜窘游,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,955評論 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望跳纳。 院中可真熱鬧忍饰,春花似錦、人聲如沸棒旗。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,770評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽铣揉。三九已至饶深,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間逛拱,已是汗流浹背敌厘。 一陣腳步聲響...
    開封第一講書人閱讀 32,000評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留朽合,地道東北人俱两。 一個(gè)月前我還...
    沈念sama閱讀 46,394評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像曹步,于是被迫代替她去往敵國和親宪彩。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,562評論 2 349

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