●BZOJ 1042 [HAOI2008]硬币购物

题链:

http://www.lydsy.com/JudgeOnline/problem.php?id=1042

题解:

容斥原理,dp预处理
首先跑个无限物品的背包dp求出dp[i]表示在四种物品都有无限个情况下有多少种方法支付 i元。
然后对于每个询问,答案就是 dp[S]-不合法的方法。
那么这个不合法的方法数怎么求呢?
举个例子:如果 c1不能超过d1个的话,那么我们就强制用掉 d1+1个 c1硬币,
那么dp[S-(d1+1)*c1]就是c1不合法的方法数。

所以这样就可以类似的求出其它硬币的不合法的方法数,以及某几种硬币都不合法的方法数,用于容斥计算。
即 ANS=dp[S] - 一种硬币不合法 + 两种硬币不合法 -三种硬币不合法 +四种硬币不合法。
DFS实现

代码:

#include<cstdio>
#include<cstring>
#include<iostream>
#define MAXN 105000
#define ll long long
#define filein(x) freopen(#x".in","r",stdin);
#define fileout(x) freopen(#x".out","w",stdout);
using namespace std;
ll dp[MAXN],c[10],d[10];
ll tot,ANS,S;
void dfs(int p,int num,ll de){
	if(p==5) return;
	ll nde=de+(d[p]+1)*c[p];
	ll val=S-nde<0?0:dp[S-nde];
	ANS+=val*(((num+1)&1)?-1:1);
	dfs(p+1,num+1,nde);
	dfs(p+1,num,de);
}
int main()
{
	dp[0]=1;
	for(int i=1;i<=4;i++) {
		scanf("%lld",&c[i]);
		for(int j=c[i];j<=100000;j++)
			dp[j]+=dp[j-c[i]];
	}
	scanf("%lld",&tot);
	while(tot--){
		for(int i=1;i<=4;i++) 
			scanf("%lld",&d[i]);
		scanf("%lld",&S);
		ANS=dp[S];
		dfs(1,0,0);
		printf("%lld\n",ANS);
	}
	return 0;
}

转载于:https://www.cnblogs.com/zj75211/p/8029358.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值