「每日LeetCode」2020年10月11日
本文最后更新于:2023年3月19日 晚上
Lt416. 分割等和子集,动态规划
416. 分割等和子集
给定一个只包含正整数的非空数组。是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。
注意:
- 每个数组中的元素不会超过 100
- 数组的大小不会超过 200
示例 1:
1 |
|
示例 2:
1 |
|
思路
动态规划-二维矩阵
具体过程可以参考官方题解分割等和子集
- 如果数组长度小于等于 1,直接返回 false
- 求数组的和及数组内元素的最大值,求和完毕,求出 target=sum/2,将问题转化为求数组中是否存在一个子集的和为 target
- 如果 target 不能整除 2,则不存在结果,返回 false
- 如果 target 比数组内元素的最大值小,则不存在结果,返回 false
- 设置一个二维矩阵,用于动态规划,问题现在变成了0-1 背包问题,横坐标为 target 值,纵坐标为原数组下标的取值范围(可以为 0 个)
- 首先将第一列,即 target=0,不管哪一行都可以取 0 个得到 target=0,所以第一列都设为 true
- 对第一行来说,即只有 nums[0]这一个元素,target 只能取到 0 或者 nums[0]本身,所以将
dp[0][nums[0]]
设为 true - 遍历剩余的行及元素,从第二行开始,相当于背包问题添加了新的物品再来判断情况,及原数组的第二个元素,之后的每行都设当前的元素为
num=nums[i]
,再对每一列 j,即能否取到当前列的值进行判断:- 如果 num>j,则说明不能取当前的元素,能否取到当前 j 的值,只看前一行对应的位置的值,直接令
dp[i][j]= dp[i-1][j]
即可。 - 如果 num<=j,这说明可以取到当前的元素,能否取到当前 j 的值,取决于前一行对应的位置是否可取或者取了当前数值以后,剩下的值在前一行是否可取,所以对应的表达式为
dp[i][j]= dp[i-1][j] || dp[i-1][j-num]
- 如果 num>j,则说明不能取当前的元素,能否取到当前 j 的值,只看前一行对应的位置的值,直接令
- dp 矩阵生成完毕以后,直接返回矩阵右下角的值即可,右下角的值即意味着转化后的问题数组中是否存在一个子集的和为 target
动态规划-一维矩阵优化
具体过程可以参考官方题解分割等和子集
- 如果数组长度小于等于 1,直接返回 false
- 求数组的和及数组内元素的最大值,求和完毕,求出 target=sum/2,将问题转化为求数组中是否存在一个子集的和为 target
- 如果 target 不能整除 2,则不存在结果,返回 false
- 如果 target 比数组内元素的最大值小,则不存在结果,返回 false
- 由之前可以看到,当前行的情况只取决于上一行,所以可以对矩阵进行优化,简化为一维矩阵,从尾部开始判断,就不会产生影响
- 对于任何元素都可取 0 个令 target=0,所以先令
dp[0]=true
- 遍历每个元素,对于每个元素来说,都从 dp 的最后一个开始判断,到当前元素的值为止即可,小于当前元素的值是不会发生变化的
- 对于任何元素都可取 0 个令 target=0,所以先令
- dp 矩阵生成完毕以后,直接返回矩阵最后一个元素的值即可
解答
动态规划-二维矩阵
1 |
|
动态规划-一维矩阵优化
1 |
|
本博客所有文章除特别声明外,均采用 CC BY-SA 4.0 协议 ,转载请注明出处!