版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/sugar_free_mint/article/details/82690436
题目
种颜色,每种颜色有 枝花,现挑出 朵,使没有颜色完全相同的方案
分析
可以发现,这道题是求多重集的组合数,根据容斥原理也就是
关于优化的方面,因为选择的数量特别大,所以说需要用二进制优化,还是比较简单去想的,对于判断越界可以用lucas定理 @my blog古代猪文,关于组合数的求法可以用乘法逆元
代码
#include <cstdio>
#define rr register
#define mod 1000000007
long long m,a[20],ans; int n,inv[20];
inline int ksm(int x,int y){//快速幂
int ans=1;
while (y){
if (y&1) ans=(long long)ans*x%mod;
x=(long long)x*x%mod; y>>=1;
}
return ans;
}
inline int c(long long n,int m){
if (n<0||m<0||n<m) return 0;//不可能存在答案
if (!n||!m) return 1;//特判
int ans=1;
for (rr int i=0;i<m;++i)
ans=(long long)ans*(n-i)%mod*inv[i]%mod;//求组合数
return ans;
}
int main(){
scanf("%d%lld",&n,&m);
for (rr int i=0;i<n;++i) scanf("%lld",&a[i]),inv[i]=ksm(i+1,mod-2);//乘法逆元
for (rr int x=0;x<1<<n;++x){
if (!x) ans=(ans+c((n+m-1)%mod,n-1))%mod;//不考虑重复的状况
else{
long long t=n+m; int p=0;
for (rr int i=0;i<n;++i)
if (x>>i&1) p++,t-=a[i];//记录1的个数
t-=p+1;
if (p&1) ans=(ans-c(t%mod,n-1))%mod;//求答案
else ans=(ans+c(t%mod,n-1))%mod;//可能要加回去(容斥定理)
}
}
printf("%lld",(ans+mod)%mod);//算下来可能会出现负数
return 0;
}