动态规划之钢条切割

问题导入

假定某公司购买长钢条,将其切割为短钢条进行出售(切割工序本身没有成本支出)。考虑求出最佳的切割方案。

出售一段长度为i的钢条价格如下表:

长度i 价格p
1 1
2 5
3 8
4 9
5 10
6 17
7 17
8 20
9 24
10 30

时间复杂度

长度为n的钢条共有2的n-1次方种不同的切割方案,因为在距离钢条左端i处,总是可以选择切割或不切割。所以普通方法(递归)解决此问题的时间复杂度为2的指数级别。

普通方法效率很低,是因为它反复求解相同的子问题

动态规划则仔细安排求解顺序,对每个子问题只求解一次,并将结果保存下来。如果随后再次需要求解此子问题的解,只需查找保存的结果,而不必重新计算。因此,动态规划是付出额外的内存空间来节省计算时间。而时间上的节省可能是非常巨大的。

动态规划原理

动态规划与分治方法相似,都是通过组合子问题的解来求解原问题。

适合使用动态规划方法求解的最优化问题应该具备两个要素:

1.最优子结构:如果一个问题的最优解包含其子问题的最优解,就称此问题具有最优子结构轻型纸。

2.子问题重叠:问题的递归算法会反复地求解相同的子问题,而不是一直产生新的子问题。

问题求解

public class Cut {
    public static void main(String[] args) {
        //钢条价格数组
        int[] p = {1,5,8,9,10,17,17,20,24,30};
        method(p, 10);
    }
    public static void method(int[] p, int n){
        //结果数组
        int[] r = new int[n+1];
        //切割方式数组
        int[] s = new int[n+1];
        for(int j = 1;j <= n;j++){
            int q = -1;     //此处-1的作用是"无穷小"
            for(int i = 1;i <= j;i++){
                if(q < p[i-1]+r[j-i]){
                    q = p[i-1]+r[j-i];
                    s[j] = i;   //表示长度为j时,被切割后的左部分长度为i,而右部分如何切割则递归查找s数组
                    r[j] = q;   //长度为j时的最大收益值
                }
            }
        }
        for (int count = 0;count <= n;count++){
            System.out.println(r[count]);
            System.out.println(s[count]);
        }
    }
}

算法主体部分为两层for循环,因此用动态规划解决此问题的时间复杂度为n的平方。

posted @ 2020-01-12 20:27  tianqibucuo  阅读(506)  评论(0编辑  收藏  举报