题目:
给你一个二进制字符串数组 strs 和两个整数 m 和 n 。
请你找出并返回 strs 的最大子集的大小,该子集中 最多 有 m 个 0 和 n 个 1。
如果 x 的所有元素也是 y 的元素,集合 x 是集合 y 的 子集 。
示例 1:
输入:strs = [“10”, “0001”, “111001”, “1”, “0”], m = 5, n = 3
输出:4
解释:最多有 5 个 0 和 3 个 1 的最大子集是 {“10”,“0001”,“1”,“0”} ,因此答案是 4 。
其他满足题意但较小的子集包括 {“0001”,“1”} 和 {“10”,“1”,“0”} 。{“111001”} 不满足题意,因为它含 4 个 1 ,大于 n 的值 3 。
class Solution {
/*
本题是等价为 01背包问题,只不过这个背包是有两个维度来决定的,一个是m, 一个是n
数组中不同长度的字符串就是不同大小的物品。
动规五部曲:
1、确定 dp 数组以及下标的含义
dp[i][j]:最多有 i 个 0 和 j 个 1 的 strs 的最大子集的大小为 dp[i][j]
2、确定递推公式
dp[i][j]可以由前一个strs里的字符串推导出来,strs里的字符串有zeroNum个0,oneNum个1。
字符串的zeroNum和oneNum相当于物品的重量(weight[i])
字符串本身的个数相当于物品的价值(value[i])
dp[i][j] = max(dp[i][j], dp[i - zeroNum][j - oneNum] + 1);
3、dp 数组如何初始化
01 背包的dp数组初始化为0即可
4、确定遍历顺序
物品就是 strs 中的字符串,
背包容量就是 m 和 n
外层for循环遍历物品,内层for循环遍历背包容量,且内层是倒叙遍历
*/
public:
int findMaxForm(vector<string>& strs, int m, int n)
{
vector<vector<int>> dp(m + 1, vector<int>(n + 1, 0));
// 一个一个字符串操作啊~
// strs 相当于物品
for(string str : strs)
{
int oneNum = 0, zeroNum = 0;
for(char c : str)
{
if(c == '0') zeroNum++;
else oneNum++;
}
// 这里的 zeroNum 和 oneNum 相当于物品的重量(weight[i]),倒序遍历
for(int i = m; i >= zeroNum; i--)
{
for(int j = n; j >= oneNum; j--)
{
// 这里为什么是 + 1,对比 01 背包问题,这里的 + 1其实就是多了1个子集
dp[i][j] = max(dp[i][j], dp[i - zeroNum][j - oneNum] + 1);
// cout << dp[i][j] << " ";
}
}
// cout << endl;
}
return dp[m][n];
}
};
本文介绍了一种解决特定二进制字符串数组问题的方法,通过动态规划算法找到最多包含m个0和n个1的最大子集。文章详细阐述了算法的实现过程,并提供了完整的代码示例。

438

被折叠的 条评论
为什么被折叠?



