Java實現動態規劃法求解0/1揹包問題

     摘要: 使用動態規劃法求解0/1揹包問題。java

     難度: 初級 算法

 

      0/1揹包問題的動態規劃法求解,前人之述備矣,這裏所作的工做,不過是本身根據理解實現了一遍,主要目的仍是鍛鍊思惟和編程能力,同時,也是爲了增進對動態規劃法機制的理解和掌握。 編程

      值得說起的一個問題是,在用 JAVA 實現時, 是按算法模型建模,仍是用對象模型建模呢? 若是用算法模型,那麼 揹包的值、重量就直接存入二個數組裏;若是用對象模型,則要對揹包以及揹包問題進行對象建模。思來想去,仍是採用了對象模型,儘管內心感受算法模型彷佛更好一些。有時確實就是這樣,對象模型雖然如今很主流,但也不是萬能的,採用其它的模型和視角,或許能夠獲得更好的解法。數組

 

     揹包建模:     測試

package algorithm.dynamicplan;
public class Knapsack {
    
    /** 揹包重量  */
    private int weight;
    
    /** 揹包物品價值  */
    private int value;
    /***
     * 構造器
     */
    public Knapsack(int weight, int value) {
        this.value = value;
        this.weight = weight;
    }
    public int getWeight() {
        return weight;
    }
    
    public int getValue() {
        return value;
    }
    
    public String toString() {
        return "[weight: " + weight + " " + "value: " + value + "]";  
    }
} 

 

  揹包問題求解:優化

      

/**
 * 求解揹包問題:
 * 給定 n 個揹包,其重量分別爲 w1,w2,……,wn, 價值分別爲 v1,v2,……,vn
 * 要放入總承重爲 totalWeight 的箱子中, 
 * 求可放入箱子的揹包價值總和的最大值。
 * 
 * NOTE: 使用動態規劃法求解 揹包問題
 * 設 前 n 個揹包,總承重爲 j 的最優值爲 v[n,j], 最優解揹包組成爲 b[n];
 * 求解最優值:
 * 1. 若 j < wn, 則 : v[n,j] = v[n-1,j];
 * 2. 若  j >= wn, 則:v[n,j] = max{v[n-1,j], vn + v[n-1,j-wn]}。
 * 
 * 求解最優揹包組成:
 * 1. 若 v[n,j] > v[n-1,j] 則 揹包 n 被選擇放入 b[n], 
 * 2. 接着求解前 n-1 個揹包放入 j-wn 的總承重中, 
 *    因而應當判斷 v[n-1, j-wn] VS v[n-2,j-wn], 決定 揹包 n-1 是否被選擇。
 * 3. 依次逆推,直至總承重爲零。
 *    
 *    重點: 掌握使用動態規劃法求解問題的分析方法和實現思想。
 *    分析方法: 問題實例 P(n) 的最優解S(n) 蘊含 問題實例 P(n-1) 的最優解S(n-1);
 *              在S(n-1)的基礎上構造 S(n) 
 *    實現思想: 自底向上的迭代求解 和 基於記憶功能的自頂向下遞歸
 */
package algorithm.dynamicplan;
import java.util.ArrayList;
public class KnapsackProblem {
    
    /** 指定揹包 */
    private Knapsack[] bags;
    
    /** 總承重  */
    private int totalWeight;
    
    /** 給定揹包數量  */
    private int n;
    
    /** 前 n 個揹包,總承重爲 totalWeight 的最優值矩陣  */
    private int[][] bestValues;
    
    /** 前 n 個揹包,總承重爲 totalWeight 的最優值 */
    private int bestValue;
    
    /** 前 n 個揹包,總承重爲 totalWeight 的最優解的物品組成 */
    private ArrayList<Knapsack> bestSolution;
    
    public KnapsackProblem(Knapsack[] bags, int totalWeight) {
        this.bags = bags;
        this.totalWeight = totalWeight;
        this.n = bags.length;
        if (bestValues == null) {
            bestValues = new int[n+1][totalWeight+1];
        }
    }
    
    /**
     * 求解前 n 個揹包、給定總承重爲 totalWeight 下的揹包問題
     * 
     */
    public void solve() {
        
        System.out.println("給定揹包:");
        for(Knapsack b: bags) {
            System.out.println(b);
        }
        System.out.println("給定總承重: " + totalWeight);
        
        // 求解最優值
        for (int j = 0; j <= totalWeight; j++) {
            for (int i = 0; i <= n; i++) {
            
                if (i == 0 || j == 0) {
                    bestValues[i][j] = 0;
                }    
                else 
                {
                    // 若是第 i 個揹包重量大於總承重,則最優解存在於前 i-1 個揹包中,
                    // 注意:第 i 個揹包是 bags[i-1]
                    if (j < bags[i-1].getWeight()) {
                        bestValues[i][j] = bestValues[i-1][j];
                    }    
                    else 
                    {
                        // 若是第 i 個揹包不大於總承重,則最優解要麼是包含第 i 個揹包的最優解,
                        // 要麼是不包含第 i 個揹包的最優解, 取二者最大值,這裏採用了分類討論法
                        // 第 i 個揹包的重量 iweight 和價值 ivalue
                        int iweight = bags[i-1].getWeight();
                        int ivalue = bags[i-1].getValue();
                        bestValues[i][j] = 
                            Math.max(bestValues[i-1][j], ivalue + bestValues[i-1][j-iweight]);        
                    } // else
                } //else         
           } //for
        } //for
        
        // 求解揹包組成
        if (bestSolution == null) {
            bestSolution = new ArrayList<Knapsack>();
        }
        int tempWeight = totalWeight;
        for (int i=n; i >= 1; i--) {
           if (bestValues[i][tempWeight] > bestValues[i-1][tempWeight]) {
               bestSolution.add(bags[i-1]);  // bags[i-1] 表示第 i 個揹包
               tempWeight -= bags[i-1].getWeight();
           }
           if (tempWeight == 0) { break; }
        }
        bestValue = bestValues[n][totalWeight];
       }
    
    /**
     * 得到前  n 個揹包, 總承重爲 totalWeight 的揹包問題的最優解值
     * 調用條件: 必須先調用 solve 方法
     * 
     */
    public int getBestValue() {    
        return bestValue;
    }
    
    /**
     * 得到前  n 個揹包, 總承重爲 totalWeight 的揹包問題的最優解值矩陣
     * 調用條件: 必須先調用 solve 方法
     * 
     */
    public int[][] getBestValues() {
        
        return bestValues;
    }
    
    /**
     * 得到前  n 個揹包, 總承重爲 totalWeight 的揹包問題的最優解值矩陣
     * 調用條件: 必須先調用 solve 方法
     * 
     */
    public ArrayList<Knapsack> getBestSolution() {
        return bestSolution;
    }
    
}

 

  揹包問題測試:this

      

package algorithm.dynamicplan;

public class KnapsackTest {
    
    public static void main(String[] args) {
        
        Knapsack[] bags = new Knapsack[] {
                new Knapsack(2,13), new Knapsack(1,10),
                new Knapsack(3,24), new Knapsack(2,15),
                new Knapsack(4,28), new Knapsack(5,33),
                new Knapsack(3,20), new Knapsack(1, 8)
        };
        int totalWeight = 12;
        KnapsackProblem kp = new KnapsackProblem(bags, totalWeight);
        
        kp.solve();
        System.out.println(" -------- 該揹包問題實例的解: --------- ");
        System.out.println("最優值:" + kp.getBestValue());    
        System.out.println("最優解【選取的揹包】: ");
        System.out.println(kp.getBestSolution());
        System.out.println("最優值矩陣:");
        int[][] bestValues = kp.getBestValues();
        for (int i=0; i < bestValues.length; i++) {
            for (int j=0; j < bestValues[i].length; j++) {
                System.out.printf("%-5d", bestValues[i][j]);
            }
            System.out.println();
        }
    }
} 

  

動態規劃法總結:spa

1. 動態規劃法用於求解非最優化問題:code

當問題實例P(n)的解由子問題實例的解構成時,好比 P(n) = P(n-1) + P(n-2) [斐波那契數列] ,而 P(n-1) 和 P(n-2)可能包含重合的子問題,可使用動態規劃法,經過自底向上的迭代,求解較小子問題實例的解,並做爲求解較大子問題實例的解的基礎。關鍵思想是: 避免對子問題重複求解。對象

好比: 求斐波那契數 F(5):

F(5)  = F(4) + F(3);

子問題: F(4) = F(3) + F(2) ;

                        F(3) = F(2) + F(1);

                                  F(2) = F(1) + F(0)

                        F(2) = F(1) + F(0);

子問題: F(3) = F(2) + F(1)

                        F(2) = F(1) + F(0)

由上面的計算過程可知,若是單純使用遞歸式,則子問題 F(2) 被重複計算了2次;當問題實例較大時,這些重複的子問題求解就會耗費大量沒必要要的時間。 若使用動態規劃法,將 F(2) 的值存儲起來,當後續計算須要的時候,直接取出來, 就能夠節省很多時間。

 

另外一個比較典型的例子是: 求解二項式係數  C(n, k) = C(n-1, k) + C(n-1, k-1)

 

2. 動態規劃法求解最優化問題:

      當問題實例P(n) 的最優解 能夠從 問題實例 P(n-1) 的最優解 構造出來時,能夠採用動態規劃法,一步步地構造最優解。

      關鍵是掌握動態規劃法求解問題時的分析方法,如何從問題導出 解的遞推式。 實際上,當導出揹包問題的遞歸式後,後來的工做就簡單多了,如何分析揹包問題,導出其最優解的遞推式,我以爲,這纔是最關鍵的地方!問題分析很重要!

相關文章
相關標籤/搜索