bzoj 3027 [Ceoi2004]Sweet——生成函数

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3027

化式子到 ( \mul_{i=1}^{n}(1-x^(m[i]+1)) ) / (1-x)^n 之后就不会了。

其实把分子拿出来后的部分可以展开成一个式子,用组合意义可知 k 次项系数是 C( n-1+k,n-1 ) 。

分子的那部分可以暴搜 2^n 种可能的项!一个项 k * x^y 对答案的贡献就是 k*( \sum_{i=L-y}^{R-y}C(n-1+i,n-1) );考虑完这 2^n 种情况对答案的贡献后答案就算好了。

组合数一列的求和可以是那个右下角位置的值。

模数可能让组合数不能除,但可以把要除的 n! 乘进模数里,即 % (mod*n!) ,最后就可以把答案除以 n! 再输出了。

注意负数。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
using namespace std;
const int N=15,M=1030;
int n,m,w[N],L,R;
ll mod,ans;
void upd(ll &x){x>=mod?x-=mod:0;}
ll calc(int k)
{
  ll ret=1;
  for(int i=k+1;i<=k+n;i++)
    ret=ret*i%mod;
  return ret;
}
void dfs(int cr,int xs,int cs)
{
  if(cs>R)return;
  if(cr>n)
    {
      ll d=calc(R-cs)+mod-(L-cs-1<0?0:calc(L-cs-1));
      upd(d);
      ans=(ans+xs*d)%mod;//xs may <0 so ans may <0!!!
      return;
    }
  dfs(cr+1,xs,cs);
  dfs(cr+1,-xs,cs+w[cr]+1);
}
int main()
{
  scanf("%d%d%d",&n,&L,&R);
  for(int i=1;i<=n;i++)scanf("%d",&w[i]);
  m=1;for(int i=1;i<=n;i++)m*=i; mod=(ll)2004*m;
  dfs(1,1,0); if(ans<0)ans+=mod;///
  printf("%lld\n",ans/m);
  return 0;
}

 

posted on 2018-11-27 21:26  Narh  阅读(286)  评论(0编辑  收藏  举报

导航