每日OJ_牛客WY15 幸运的袋子

目录

牛客WY15 幸运的袋子

题解代码


牛客WY15 幸运的袋子


题解代码

        本题的本质是求符合条件的子集个数。 每次从全集中选择若干元素(小球)组成子集(袋子)。对于任意两个正整数a,b如果满足 a+b>ab,则必有一个数为1.可用数论证明:设a=1+x,b=1+y,则1+x+1+y>(1+x)(1+y),---> 1>xy,则x,y必有一个为0,即a,b 有一个为1.推广到任意k个正整数,假设a1,a2,...ak,如果不满足给定条件,即和sum小于等于积pi。如果此时再选择一个数b,能使其满足sum+b > pib,则,b必然为1,且为必要非充分条件。反之,如果选择的b>1,则 sum+b <= pi*b,即a1,a2,...,ak,b不满足给定条件。

        因此,将球按标号升序排序。每次从小到大选择,当选择到a1,a2,...,ak-1时满足给定条件,而再增加选择ak 时不满足条件(ak必然大于等于max(a1,a2,...,ak-1)),继续向后选择更大的数,必然无法满足!此时不必再继续向后搜索。如果有多个1,即当k=1时,sum(1)>pi(1)不满足,但下一个元素仍为1,则可以满足 1+1>1*1, 所以要判断当前ak是否等于1,如果等于1,虽然不能满足,组合的个数不能增加,但是继续向后搜索,仍然有满足条件的可能.对于重复数字,组合只能算一个,要去重。

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;

int getLuckyBag(const vector<int>& v, int pos, int sum, int mul)
{
	int count = 0;
	for (int i = pos; i < v.size(); i++)
	{
		sum += v[i];
		mul *= v[i];
		if (sum > mul) //只有数据有1的时候才可能sum>mul
			count += 1 + getLuckyBag(v, i + 1, sum, mul);
		else if (v[i] == 1) // 如果不幸运,并且当前值为1(数据只有1才会这样)
			count += getLuckyBag(v, i + 1, sum, mul);
		else
			break;
		sum -= v[i];
		mul /= v[i];
		while (i < v.size() - 1 && v[i] == v[i + 1])
		{	// 处理重复问题
			i++;
		}
	}
	return count;
}

int main()
{
	int n = 0;
	cin >> n;
	vector<int> v;
	v.reserve(n);
	for (int i = 0;i < n;i++)
	{
		int j = 0;
		cin >> j;
		v.push_back(j);
	}
	sort(v.begin(), v.end());
	cout << getLuckyBag(v, 0, 0, 1);
}
  • 4
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

GR鲸鱼

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值