LeetCode 416. 分割等和子集(bitset优化)
LeetCode 416. 分割等和子集
1 题目描述
给你一个只包含正整数的非空数组nums。请你判断是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。
1.1 输入测试
示例 1:
输入:nums = [1,5,11,5]
输出:true
解释:数组可以分割成 [1, 5, 5] 和 [11] 。
示例 2:
输入:nums = [1,2,3,5]
输出:false
解释:数组不能分割成两个元素和相等的子集。
1.2 数据范围
1 <= nums.length <= 200
1 <= nums[i] <= 100
2 解题思路
将数组分割成两个元素和相等的子集,意味着这两个子集的元素和都是总元素和sum的一半。
所以,道题的目标就是判断是否存在元素和为sum/2的子集。
其中包含的一个特殊情况是,当sum为奇数时,结果明显是不存在的。
2.1 DFS枚举
初看数据量比较小,于是就尝试DFS暴力枚举,最后果不其然超时了。
class Solution {
public:
bool canPartition(vector<int>& nums) {
int tar = 0;
for (auto &num : nums)
{
tar += num;
}
if (tar & 1)
{
return false;
}
return dfs(nums, 0, 0, tar >> 1);
}
bool dfs(vector<int>& nums, int i, int sum, int tar)
{
if (sum == tar)
{
return true;
}
if (sum > tar)
{
return false;
}
return i == nums.size() ? false : dfs(nums, i + 1, sum + nums[i], tar) || dfs(nums, i + 1, sum, tar);
}
};
2.2 记忆化搜索
既然暴力枚举不行,那就加上记忆化搜索,通过记录相同sum和i时的结果,达到减少重复计算的目的。
最终成功通过,运行时间只有55ms,但其实还有更优的解法。
class Solution {
public:
bool canPartition(vector<int>& nums) {
int tar = 0;
for (auto &num : nums)
{
tar += num;
}
if (tar & 1)
{
return false;
}
return dfs(nums, 0, 0, tar >> 1);
}
bool vis[10001][201]{};
bool mem[10001][201]{};
bool dfs(vector<int>& nums, int i, int sum, int tar)
{
if (sum == tar)
{
return true;
}
if (sum > tar)
{
return false;
}
if (vis[sum][i])
{
return mem[sum][i];
}
vis[sum][i] = true;
return mem[sum][i] = i == nums.size() ? false : dfs(nums, i + 1, sum + nums[i], tar) || dfs(nums, i + 1, sum, tar);
}
};
2.3 DP
重新观察上面的计算过程就能发现,实际上我们所做的就是在枚举所有可能的子集和,这其实就是典型的0/1背包问题。
因此,我们可以采用动态规划的思想,构造一个初始只包含一个数字0的子集和数组v,接着依次从nums集合中取出一个数,与数组v中的子集和逐个相加得到一批新的子集和,并添加进子集和数组v。直到求得目标结果,或是所有数字都被取出。
这种写法的运行时间仅为15ms,但接下来我们还要介绍另一种DP的写法,为后面的bitset优化作铺垫。
class Solution {
public:
bool canPartition(vector<int>& nums) {
int tar = 0;
vector<int> v{0};
bool vis[10001]{};
for (auto &num : nums)
{
tar += num;
}
if (tar & 1)
{
return false;
}
tar >>= 1;
for (auto &num : nums)
{
for (int i = v.size() - 1; i >= 0; i--)
{
int n = num + v[i];
if (n == tar)
{
return true;
}
if (n > tar || vis[n])
{
continue;
}
vis[n] = true;
v.push_back(n);
}
}
return false;
}
};
2.4 DP + bitset
在上面的DP写法中,我们使用了数组v记录已知的子集和,但也可以看出vis数组同时也记录了对应下标值的子集和是否存在。
因此,我们可以不使用v数组,而是通过遍历vis数组的方式来判断子集和是否存在。当然,根据0/1背包的特性,遍历的过程需要从后往前。
最终运行时间为23ms,比第一个DP方法稍慢,毕竟我们是遍历范围内所有可能的下标,而不像之前有额外的记录可以快速访问,但这么做也能相应地减少内存消耗。
不过这并不是我们的最终目标。
class Solution {
public:
bool canPartition(vector<int>& nums) {
int tar = 0;
bool vis[10001]{1};
for (auto &num : nums)
{
tar += num;
}
if (tar & 1)
{
return false;
}
tar >>= 1;
for (auto &num : nums)
{
for (int i = tar; i >= num; i--)
{
// 等价于 vis[i] |= vis[i - num];
// if (vis[i - num])
// {
// vis[i] = true;
// }
vis[i] |= vis[i - num];
}
}
return vis[tar];
}
};
其实从上面的内层for循环中可以看出,我们每次所做的操作都是将第i位元素与第i偏移num位的元素进行或运算。所以,我们可以把内层循环的结果等价为,对整个vis数组做移位或运算,也就是vis |= vis << num
但即使我们知道这样的等价操作,也没办法直接对整个bool数组进行移位,这时候就需要用到支持<<
、>>
位移操作的bitset容器来实现了。
其使用方法也很简单,只需要将原来的bool vis[N]
替换成bitset<N> vis
即可。
于是,我们的运行时间成功缩短到了5ms。对比第一种DP算法,这种方法在减少内存占用的同时也缩短了运行时间。
class Solution {
public:
bool canPartition(vector<int>& nums) {
int tar = 0;
bitset<10001> vis{1};
for (auto &num : nums)
{
tar += num;
}
if (tar & 1)
{
return false;
}
tar >>= 1;
for (auto &num : nums)
{
vis |= vis << num;
}
return vis[tar];
}
};
其实到了这一步,我们还能将代码逻辑进行进一步的合并。将tar的计算合并到vis的for循环中,并在结尾统一对tar和vis的结果进行判断。
最终版本如下所示
class Solution {
public:
bool canPartition(vector<int>& nums) {
int tar = 0;
bitset<10001> vis{1};
for (auto &num : nums)
{
tar += num;
vis |= vis << num;
}
return !(tar & 1) && vis[tar >> 1];
}
};
本文发布于2024年3月26日
最后编辑于2024年3月26日
LeetCode 416. 分割等和子集(bitset优化)的更多相关文章
- Leetcode 416分割等和子集
416. 分割等和子集 已知是个背包问题,由于可以等分为两部分,所以必定是个偶数. 一开始想到的是回溯法 bool helper(vector<int>&nums, int i, ...
- Java实现 LeetCode 416 分割等和子集
416. 分割等和子集 给定一个只包含正整数的非空数组.是否可以将这个数组分割成两个子集,使得两个子集的元素和相等. 注意: 每个数组中的元素不会超过 100 数组的大小不会超过 200 示例 1: ...
- Leetcode 416.分割等和子集
分割等和子集 给定一个只包含正整数的非空数组.是否可以将这个数组分割成两个子集,使得两个子集的元素和相等. 注意: 每个数组中的元素不会超过 100 数组的大小不会超过 200 示例 1: 输入: [ ...
- [LeetCode]494. 目标和、416. 分割等和子集(0-1背包,DP)
题目一 494. 目标和 给定一个非负整数数组,a1, a2, ..., an, 和一个目标数,S.现在你有两个符号 + 和 -.对于数组中的任意一个整数,你都可以从 + 或 -中选择一个符号添加在前 ...
- 【LeetCode动态规划#06】分割等和子集(01背包问题一维写法实战)
分割等和子集 分割等和子集 给你一个 只包含正整数 的 非空 数组 nums .请你判断是否可以将这个数组分割成两个子集,使得两个子集的元素和相等. 示例 1: 输入:nums = [1,5,11,5 ...
- AtCoder Regular Contest 070 D - No Need 想法:利用单调性二分+bitset优化
/** 题目:D - No Need 链接:http://arc070.contest.atcoder.jp/tasks/arc070_b 题意:给出N个数,从中选出一个子集,若子集和大于等于K,则这 ...
- 洛谷 P5527 - [Ynoi2012] NOIP2016 人生巅峰(抽屉原理+bitset 优化背包)
洛谷题面传送门 一道挺有意思的题,想到了某一步就很简单,想不到就很毒瘤( 首先看到这样的设问我们显然可以想到背包,具体来说题目等价于对于每个满足 \(i\in[l,r]\) 的 \(a_i\) 赋上一 ...
- hdu 5506 GT and set dfs+bitset优化
GT and set Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) Probl ...
- hdu 5745 La Vie en rose DP + bitset优化
http://acm.hdu.edu.cn/showproblem.php?pid=5745 这题好劲爆啊.dp容易想,但是要bitset优化,就想不到了. 先放一个tle的dp.复杂度O(n * m ...
- hdu_5036_Explosion(bitset优化传递闭包)
题目链接:hdu_5036_Explosion 题意: 一个人要打开或者用炸弹砸开所有的门,每个门里面有一些钥匙,一个钥匙对应一个门,有了一个门的钥匙就能打开相应的门,告诉每个门里面有哪些门的钥匙,问 ...
随机推荐
- String - 一些测试(持续更新)
void main() { char *buffer = new char(1000); memset(buffer, 0, 1000); char buffer1[1000] = {}; buffe ...
- 【Android 逆向】【攻防世界】android2.0
这是一道纯算法还原题 1. apk安装到手机,提示输入flag,看来输入就是flag 2. jadx 打开apk查看 this.button.setOnClickListener(new View.O ...
- Flutter——安装依赖包时,出现Waiting for another flutter command to release the startup lock
问题描述 运行 flutter packages get 时 出现 Waiting for another flutter command to release the startup lock 解决 ...
- CentOS8安装Geant4笔记(二):CentOS8安装Qt5.15.2并测试运行环境
前言 在服务器CentOs8.2上安装geant4软件,但是运行不起来,所以本节开始主要是安装qt,测试qt基本功能. 要点 添加qt环境到系统环境中,是geant4启动qt的必要条件. ...
- Kotlin 函数 与 lambda 表达式
一.函数 代码块函数体: fun sum(x: Int, y: Int): Int { return x + y } 表达式函数体: fun sum(x: Int, y: Int) = x + y 使 ...
- nodejs内存泄漏概要分析
const heapdump = require('heapdump'); setTimeout( ()=>{ heapdump.writeSnapshot(`${process.cwd()}/ ...
- 【Azure 事件中心】Event Hubs中存在非常多的错误数据,是否能提前删除这些数据呢?
问题描述 因为一些特殊原因,Event Hub 里面堆积了很多不需要的数据事件,正常要等事件中的过期时间到后才有Event Hub自动删除掉,但希望能够尽快马上删除,有没有什么手动的方法吗? 问题解答 ...
- 【Azure Redis 缓存】C#程序是否有对应的方式来优化并缩短由于 Redis 维护造成的不可访问的时间
问题描述 C#程序是否有对应的方式来优化并缩短由于 Redis 维护造成的不可访问的时间? Redis维护说明: Redis 服务维护时,会把副本节点提升为主节点,且旧主节点关闭现有连接时,这个时候, ...
- SpringCloud Sentinel使用
1. 简介 Sentinel 是面向分布式服务架构的流量控制组件,主要以流量为切入点,从限流.流量整形.熔断降级.系统负载保护.热点防护等多个维度来帮助开发者保障微服务的稳定性.替换原先Hystrix ...
- [.Net 6]写一个简单的文件上传控件后端
此项目是配合上一篇文章[Vue]写一个简单的文件上传控件 - 林晓lx - 博客园 (cnblogs.com) 的后端程序,使用.Net 6项目框架搭建,开发前请安装Visual Studio 20 ...