bzoj2013[CEOI2010] A huge tower

题意

有N(2<=N<=620000)快砖,要搭一个N层的塔,要求:如果砖A恰好在砖B上面,那么A不能比B的长度+D要长。问有几种方法,输出 答案 mod 1000000009的值

分析

没想出来
菜鸡.jpg
考虑按照砖的长度从短到长依次加入.
对于一个合法的塔,我们抽掉其中最长的一块砖,剩下的砖按照原先的顺序必然还是合法的塔.

假如最长的砖在最下方,那么显然不会从合法变成不合法.
假如不在最下方,设最长的砖长度为x,它下方的砖长度为a,上方的砖长度为b.
那么a+d>=x.抽掉x之后,因为x>=b,所以必然a+d>=b

那么我们现在在一个合法的塔中插入一个比现在所有砖都长的砖.因为比现在的所有砖都长,这块新加进来的砖一定不会和它上方的砖产生矛盾.只需要考虑哪些位置不会和它下方的砖产生矛盾即可.也就是找出有多少块砖长度>=新加的砖的长度-d.注意新加的砖还可以在整个塔最下方,这一步的方案数是"长度>=新加的砖的长度-d"的砖的块数+1.
每一步的方案数乘起来即可.

#include<cstdio>
#include<algorithm>
using namespace std;
const int maxn=700005;
const int mod=1000000009;
int a[maxn];
int main(){
  int n,d;scanf("%d%d",&n,&d);
  for(int i=1;i<=n;++i)scanf("%d",a+i);
  sort(a+1,a+n+1);
  int ans=1;
  int pt=1;
  for(int i=1;i<=n;++i){
    while(pt<=n&&a[pt]<a[i]-d)++pt;
    ans=ans*1ll*(i-pt+1)%mod;
  }
  printf("%d\n",ans);
  return 0;
}

posted @ 2017-06-14 15:34  liu_runda  阅读(579)  评论(0编辑  收藏  举报
偶然想到可以用这样的字体藏一点想说的话,可是并没有什么想说的. 现在有了:文化课好难