[BZOJ]1042 硬币购物(HAOI2008)

时间:2022-03-12 22:17:53

  失踪OJ回归。

  小C通过这道题mark一下容斥一类的问题。

Description

  硬币购物一共有4种硬币。面值分别为c1,c2,c3,c4。某人去商店买东西,去了tot次。每次带di枚ci硬币,买s的价值的东西。请问每次有多少种付款方法。

Input

  第一行 c1,c2,c3,c4,tot 下面tot行 d1,d2,d3,d4,s。

Output

  每次的方法数。

Sample Input

  1 2 5 10 2
  3 2 3 1 10
  1000 2 2 2 900

Sample Output

  4
  27

HINT

  di,s<=100000,tot<=1000。

Solution

  O(s*tot)的DP算法谁都会写,但是看着这时间复杂度你难道不虚吗?

  你难道甘愿被卡常而就此丢掉10分或是在刷题时打一个这样的暴力草草了事而对其中的精妙不闻不问,你的良心不会痛吗?

  如果你在解题时看到题目中有常数a(a<20)这样的数据,不妨就往O(2a)这样复杂度的算法去想一想。

  因为题目中有一个常数4,而且还都是限制条件,所以我们就往状压、容斥这方面去想。

  显然状压是不可能的,于是我们就只有容斥了。

  首先我们要知道容斥在这道题是干嘛用的:

  容斥就是对多个限制条件下方案的去重工作,也就是你们所熟知的,求多个集合的并集。

  容斥常常伴随的思想是一种逆向思维,就是题目往往要求我们去求多个集合的交,然而我们并没有好的办法,转而跑去求各个集合的补的并,再补回来就是各个集合的交。

  我们需要理解在这个算法中, 交 是可以O(1)求得的,然而 并 需要用容斥求得。

  所以解这样的题目的大致思路就是:

    题目要求我们求A1~An的交,但是我们发现很难求;

    所以我们去求CuA1~CuAn的并,而CuA1~CuAn的并需要我们求CuA1~CuAn的交,但是我们发现CuA1~CuAn的交特别好求,所以我们就圆满地解决了这个问题。

  运用这样的思路,我们就可以很快解出这道题。

  所以我们要求的只剩,在这次购物中,第 i 种硬币用了超过di的方案数。

  脑补一下,我们就知道,我们只要每种硬币先取到它们限制的数量+1,剩下随便取就可以了,这样的方案无论如何都是满足每种硬币都超过限制的。

  所以用一句话概括题解:容斥,求f[s-Σ(ci*(di+1))],f[x]为在没有任何限制下,取硬币得到面值x的方案数。

  时间复杂度O(max(s)*4+tot*16)。注意答案最大为C(s,4)会爆int。

#include <cstdio>
#include <cstring>
#include <algorithm>
#define ll long long
#define MM 100005
using namespace std;
int n,m;
int a[],g[],ys[];
ll f[MM],lt,ans;
bool u; inline int read()
{
int n=,f=; char c=getchar();
while (c<'' || c>'') {if(c=='-')f=-; c=getchar();}
while (c>='' && c<='') {n=n*+c-''; c=getchar();}
return n*f;
} int main()
{
register int i,j;
a[]=read(); a[]=read(); a[]=read(); a[]=read(); n=read();
f[]=ys[]=; ys[]=; ys[]=; ys[]=;
for (i=;i<=;++i)
for (j=a[i];j<MM;++j) f[j]+=f[j-a[i]];
while (n--)
{
g[]=read(); g[]=read(); g[]=read(); g[]=read(); m=read();
ans=;
for (i=;i<;++i)
{
for (u=lt=,j=;j<=;++j)
if (i&ys[j]) u^=,lt+=1LL*a[j]*(g[j]+);
if (lt>m) continue;
ans+=(f[m-lt])*(u?-:);
}
printf("%lld\n",ans);
}
}

Last Word

  大概就是小C关于容斥的一点点想法,当然这样的题目还有很多无法以偏概全。希望这样的思路能对以后有一点帮助吧。