Datawhale编程——动态规划DP

0-1背包问题

问题:有n个物品,第i个物品价值为vi,重量为wi,其中vi和wi均为非负数,背包的容量为W,W为非负数。现需要考虑如何选择装入背包的物品,使装入背包的物品总价值最大。

针对这个经典的动态规划问题,先建立一个实例,如下表格:

物体编号i 1 2 3 4 5
价值v 4 5 10 11 13
重量w 3 4 7 8 9

假设约束条件背包的最大承重\(W = 17\),且每个物体只能装一次,请问背包应该怎么装最好。

这一类问题依旧是两个关键:1.初始条件或者最小子问题最优解;2.递归定义式或状态转移方程。

这里最小子问题最优解应该就是,背包只选一个物品且在限定承重 W' 时所指向的最佳物体。这里 W' 可以为大于等于1的任意值。

那么状态转移方程应该就是,在当前给定限定承重条件下,从不同的承重组合选出一个最优解(每一个都是它在限定承重条件下得到的最优解)。用数学定义式可以如下:

\[dp[i][j] = max(dp[i-1][j], dp[i-1][j-w(k)]+v[k]) \]

其中dp[i][j]表示i件物品放入一个容量为j的背包可以获得的最大价值。这个式子不是很好理解,举个例子就很清楚了,如下。

比如当前背包容量是10,那我们可以有多种分配组合,比如9+1,以及7+3。其中dp[i-1][j]可以对应分配的9,dp[i-1][j-w(k)]可以对应分配的7;左边9所对应的1是再找不出适合装的东西了,右边7对应的3可以再装一个编号1。这个式子的意义就是,比较这两种组合其中的更佳者。

需要注意的是,不一定永远是右边的更佳,有时候空间不全部利用反而能得到更加解。这就是动态规划的关键

要是直接将动态规划的代码写出来,有时候可能很困难。可以先考虑用递归做,再将自顶向下的递归转化为自底向上的动态规划就好了。这里就不管这些,直接给出动态规划的代码了。

class ZeroByOnePack:
    def __init__(self, list_v, list_w):
        self.list_v = list_v
        self.list_w = list_w

    def dpPack(n, max_w):
        V = self.list_v
        W = self.list_w
        dp = [0] * (max_w + 1) # 为了形式上好理解,只好浪费一部分空间
        for i in range(n);
            for j in range(W[i], max_w + 1):
                dp[j] = max(dp[j], dp[j-W[i]]+V[i])
        return dp[max_w]                

这里只用一维数组,节省内存空间。

leetcode 132

代码实现

作业限制了用动态规划,但其实这道题不用动态规划也可以做。

class Solution(object):
    def minCut(self, s):
        """
        :type s: str
        :rtype: int
        """
        cut = [x for x in range(-1,len(s))]
        for i in range(0,len(s)):
            for j in range(i,len(s)):
                if s[i:j] == s[j:i:-1]:
                    cut[j+1] = min(cut[j+1],cut[i]+1)
        return cut[-1]

当然,这个性能是很差的,时间复杂度应该达到了O(n3)?如果用动态规划的话,时间复杂度只要O(n2)。

class Solution:
    def minCut(self, s):
        """
        :type s: str
        :rtype: int
        This can be solved by:
        
        cut[end] is the minimum of cut[st-1] + 1 (st <= end) if [st, end] is palindrome
        
        a   b   a   |   c    c
                        st  end
               st-1 |  [st,  end] is palindrome
          cut(st-1) +  1
          
        """
        n = len(s)
        cut = [0] * n
        pal = [[False] * n for row in range(n)]
        
        for end in range(n):
            min_cut = end
            for st in range(end + 1):
                if s[st] == s[end] and (end - st <= 2 or pal[st+1][end-1]):
                    pal[st][end] = True
                    min_cut = 0 if st == 0 else min(min_cut, cut[st-1] + 1)
            cut[end] = min_cut
        
        return cut[n-1]

代码不是自己写的。我只能根据简单的思路写出递归的算法,但是这道题卡了递归的边界。也许将递归算法改进成备忘录方法会更好点,这个留待以后验证了。

posted @ 2019-01-09 20:11  Wunsam_Chan  阅读(326)  评论(0编辑  收藏  举报