题解 | 数组划分

题目链接

题目:把含若干数字的数组任意划分成两个子数组,使得两个子数组的和相差最小。

本题采用3种解法,最后一种ac。

解法一:动态规划(数组和不大时可以采用,否则容易超时)

思路:设数组总和为sum,目标是找到一个子数组使得子数组和尽可能<=sum/2,这样两个数组和若相等,则和相差最小是0。

对于每一个数组中数字,都要更新dp数组:如果之前能凑出vec[i](数组元素),那么现在有了j,你就能凑出vev[i]+j了(前提是 vev[i]+j 不超过 target)

#include<stdio.h>
#include<vector>
#include<algorithm>
using namespace std;
int sum = 0;
int min_partition(vector<int> vec) {
	int target = sum / 2;//
	//dp[i]=true表示:和为i的数组能凑出
	vector<bool> dp(target + 1, false);
	dp[0] = true;
	//动态规划填表
	for (int i = 0; i<vec.size(); i++) {
		for (int j = target; j-vec[i] >=0; j--) {//逆着判断可以避免重复
			if (dp[j - vec[i]]) {
				dp[j] = true;
			}
		}
	}

	for (int i = target; i >= 0; i--) {
		if (dp[i]) {
			return i;//返回最接近target的和
			break;
		}
	}
	return -1;
}
int main(){
	vector<int> vec;
	int data;
	while (scanf("%d", &data)!=EOF) {
		vec.push_back(data);
		sum += data;
	}
	int a_sum = min_partition(vec);
	int b_sum = sum - a_sum;
	//降序输出两个元素和
	printf("%d %d", max(a_sum, b_sum),min(a_sum, b_sum));
	return 0;
}

解法二:贪心算法:先排序,依次把最大的数字插入和最小的数组中,快速得出近似出最优解!

#include<stdio.h>
#include<vector>
#include<algorithm>

using namespace std;
int sum = 0;
void min_partition(vector<int> &vec,int &a_sum,int &b_sum) {
	sort(vec.begin(), vec.end());
	int sub_sum = 0;
	for (int i = vec.size()-1; i>=0; i--) {//从大的数字开始,使得子数组和尽快接近总和的一半
		if (a_sum >= b_sum) {
			b_sum += vec[i];
		}
		else {
			a_sum += vec[i];
		}
	}
}
int main(){
	vector<int> vec;
	int data;
	while (scanf("%d", &data)!=EOF) {
		vec.push_back(data);
		sum += data;
	}
	int a_sum=0,b_sum= 0;
	min_partition(vec, a_sum, b_sum);
	//降序输出两个元素和
	printf("%d %d", max(a_sum, b_sum),min(a_sum, b_sum));
	return 0;
}

解法三:DSP+剪枝

思路:只要sa+vec[i]不超过数组总和的一半,就递归增加sa(保存子数组和)的值。

为了提高程序运行效率,先把数组降序排,预测更新全局最优的diff值,并且判断剪枝:如果预判发现 diff 已经是 0 或 1(理论极限),或者当前数字大到不可能产生更优解(2*vec[pos] > sum),就立刻设置 flag=true,停止一切后续搜索。

#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<vector>
#include<algorithm>
using namespace std;
int sum = 0;
int diff = 0;//差值
bool flag = false;//是否剪枝
void dfs(vector<int>& vec, int pos, int sa) {
	if (pos == vec.size()||flag)return;//遍历完数组or需要剪枝
	int newdiff;
	 newdiff=abs(2 * (sa + vec[pos]) - sum);
	if (newdiff < diff) {//更新最小差值
		diff = newdiff;
		if (diff == 0 || diff == 1 || 2*vec[pos] > sum) {
			flag = true;//剪枝优化
		}
	}
   //1.vec[pos]可放入sa时
	if ((sa + vec[pos])*2 < sum) {//sa+ vec[pos]小于sum/2
		dfs(vec, pos + 1, sa + vec[pos]);//往sa继续加数据
	}
	//2.vec[pos]不放入sa时
	dfs(vec, pos + 1, sa);
}
bool compare(int l, int  r) {
	return l > r;
}
int main(){
	vector<int> vec;
	int i;
	while (scanf("%d", &i) != EOF) {
		vec.push_back(i);
		sum += i;
	}
	diff = sum;//初始化差值上限
	sort(vec.begin(), vec.end(),compare);//降序排
	dfs(vec, 0, 0);
	int sa = (sum - diff) / 2;//diff=(sum-sa)-sa=sum-2*sa
	printf("%d %d\n", sa + diff, sa);//降序输出
    return 0;
}

计算机复试机试(王道版) 文章被收录于专栏

收录王道2026年计算机复试机试的(课程)代码题解,仅供个人学习参考

全部评论

相关推荐

当年还在美团那个倒霉的&nbsp;Peppr&nbsp;团队工作时,我一直有个疑问:这群人每天到底在自嗨什么。每次开会一堆人围着一堆“看起来很高级”的文档转,模板统一、名词复杂、页数感人,每一页都在暗示一件事:“你不懂,是因为你不专业。”但现实是——代码照样写在&nbsp;💩&nbsp;山上,该出问题还是会出问题,这真的很逗,系统一出问题,文档的唯一作用就是证明:“我们当初确实认真写过文档。”所以本质区别到底是什么?是代码质量提升了,还是大家在精神层面完成了一次“工程师&nbsp;cosplay”?有句话说得好潮水退去才知道谁在裸泳。还记得当时的马哥、明哥(图&nbsp;1&nbsp;左)最爱反复强调一句话:“所有场景一定要想到。”、“这个场景为什么没考虑到?”不过他们这些话我是真的听进去了。不然我也不会在一年多前就说:这个项目活不过两年。顺带一提,那段时间还有个固定节目。每次下楼,总能听见我明哥在吐槽不同的人。我从他身后绕过去,经常能听到他一边抽烟一边说:“xx&nbsp;这小子太坑了,回头我一定要跟马哥说说。”于是深谙人情世故但真不会抽烟的我也会从口袋掏出一支低尼古丁含量的烟给自己点上,假意自己什么都没听到什么都不知道,只是来抽烟的。后来我才明白,这可能也是团队文化的一部分:问题永远在别人身上,而我们,永远在复盘里😂。
秋招白月光
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务