题目中矩形的尺寸太大,导致墓地的数目太多,如果我们统计每一个墓地的虔诚度,超时是一定的。

而常青树的数目<=1e5.这启发我们从树的方向去思考。

考虑一行没有树的情况,显然这一行的墓地的虔诚度之和为0.也就是说我们可需要考虑常青树在的行就行了。

对于在同一行的每两颗长青树之间,墓地的虔诚度之和为C(l,k)*C(r,k)*sigma(C(up,k)*C(under,k)).这里的l是左边的这棵树的左边有多少颗树,r同理。up则是这一段的每个墓地上面有多少颗树,under同理。对于求和,我们可以用树状数组logn的时间查询。

而扫描完一颗树的时候,需要更新C(up,k)*C(under,k).因为树的数目<=1e5,所以总复杂度不过O(n*logn).

所以我们只需要预处理出组合数,然后离散化x坐标,再一行一行的利用BIT查询并更新。

这里的一个优化常数的小技巧是:由于题目要求对1<<31取模,中途过程中可以让int自然溢出,最后的答案mod((1<<31)-1)即可。

  1. # include <cstdio>
  2. # include <cstring>
  3. # include <cstdlib>
  4. # include <iostream>
  5. # include <vector>
  6. # include <queue>
  7. # include <stack>
  8. # include <map>
  9. # include <set>
  10. # include <cmath>
  11. # include <algorithm>
  12. using namespace std;
  13. # define lowbit(x) ((x)&(-x))
  14. # define pi 3.1415926535
  15. # define eps 1e-
  16. # define MOD
  17. # define INF
  18. # define mem(a,b) memset(a,b,sizeof(a))
  19. # define FOR(i,a,n) for(int i=a; i<=n; ++i)
  20. # define FO(i,a,n) for(int i=a; i<n; ++i)
  21. # define bug puts("H");
  22. # define lch p<<,l,mid
  23. # define rch p<<|,mid+,r
  24. # define mp make_pair
  25. # define pb push_back
  26. typedef pair<int,int> PII;
  27. typedef vector<int> VI;
  28. # pragma comment(linker, "/STACK:1024000000,1024000000")
  29. typedef long long LL;
  30. int Scan() {
  31. int res=, flag=;
  32. char ch;
  33. if((ch=getchar())=='-') flag=;
  34. else if(ch>=''&&ch<='') res=ch-'';
  35. while((ch=getchar())>=''&&ch<='') res=res*+(ch-'');
  36. return flag?-res:res;
  37. }
  38. void Out(int a) {
  39. if(a<) {putchar('-'); a=-a;}
  40. if(a>=) Out(a/);
  41. putchar(a%+'');
  42. }
  43. const int N=;
  44. //Code begin...
  45.  
  46. struct Node{int x, y;}node[N];
  47. int C[N][], tree[N], siz, col[N], now[N];
  48. VI v;
  49. vector<PII> vv;
  50.  
  51. bool comp(Node a, Node b){
  52. if (a.y==b.y) return a.x<b.x;
  53. else return a.y<b.y;
  54. }
  55. void init(){
  56. FOR(i,,) C[i][]=;
  57. FOR(i,,) FOR(j,,) C[i][j]=C[i-][j]+C[i-][j-];
  58. }
  59. void add(int x, int val){while (x<=siz) tree[x]+=val, x+=lowbit(x);}
  60. int query(int x){
  61. int res=;
  62. while (x) res+=tree[x], x-=lowbit(x);
  63. return res;
  64. }
  65. int cal(int x, int y){return x<y?:C[x][y];}
  66. int main ()
  67. {
  68. init();
  69. int n, m, T, k, ans=;
  70. n=Scan(); m=Scan(); T=Scan();
  71. FOR(i,,T) node[i].x=Scan(), node[i].y=Scan(), v.pb(node[i].x);
  72. k=Scan();
  73. sort(node+,node+T+,comp);
  74. sort(v.begin(),v.end());
  75. siz=unique(v.begin(),v.end())-v.begin();
  76. FOR(i,,T) {
  77. node[i].x=lower_bound(v.begin(),v.begin()+siz,node[i].x)-v.begin()+;
  78. ++col[node[i].x];
  79. }
  80. int i=;
  81. while (i<=T) {
  82. vv.clear();
  83. vv.pb(mp(node[i].x,node[i].y));
  84. ++i;
  85. while (i<=T&&node[i].y==node[i-].y) vv.pb(mp(node[i].x,node[i].y)), ++i;
  86. int ss=vv.size();
  87. if (ss>k) FOR(j,k,ss-k) {
  88. ans+=(C[j][k]*C[ss-j][k]*(query(vv[j].first-)-query(vv[j-].first)));
  89. }
  90. FO(j,,ss) {
  91. int fi=vv[j].first;
  92. add(fi,-*cal(now[fi],k)*cal(col[fi]-now[fi],k));
  93. ++now[fi];
  94. add(fi,cal(now[fi],k)*cal(col[fi]-now[fi],k));
  95. }
  96. }
  97. printf("%d\n",ans&);
  98. return ;
  99. }

BZOJ 1227 虔诚的墓主人(离散化+树状数组)的更多相关文章

  1. luogu2154 [SDOI2009] 虔诚的墓主人 离散化 树状数组 扫描线

    题目大意 公墓可以看成一块N×M的矩形,矩形的每个格点,要么种着一棵常青树,要么是一块还没有归属的墓地.一块墓地的虔诚度是指以这块墓地为中心的十字架的数目,一个十字架可以看成中间是墓地,墓地的正上.正 ...

  2. [BZOJ1227][SDOI2009]虔诚的墓主人 组合数+树状数组

    1227: [SDOI2009]虔诚的墓主人 Time Limit: 5 Sec  Memory Limit: 259 MBSubmit: 1433  Solved: 672[Submit][Stat ...

  3. BZOJ1227 [SDOI2009]虔诚的墓主人 【树状数组】

    题目 小W 是一片新造公墓的管理人.公墓可以看成一块N×M 的矩形,矩形的每个格点,要么种着一棵常青树,要么是一块还没有归属的墓地.当地的居民都是非常虔诚的基督徒,他们愿意提前为自己找一块合适墓地.为 ...

  4. 【Luogu】P2154虔诚的墓主人(树状数组)

    题目链接 这题就是考虑我们有这样一个情况

  5. bzoj1227: [SDOI2009]虔诚的墓主人(树状数组,组合数)

    传送门 首先,对于每一块墓地,如果上下左右各有$a,b,c,d$棵树,那么总的虔诚度就是$C_k^a*C_k^b*C_k^c*C_k^d$ 那么我们先把所有的点都给离散,然后按$x$为第一关键字,$y ...

  6. BZOJ1227 SDOI2009 虔诚的墓主人【树状数组+组合数】【好题】*

    BZOJ1227 SDOI2009 虔诚的墓主人 Description 小W 是一片新造公墓的管理人.公墓可以看成一块N×M 的矩形,矩形的每个格点,要么种着一棵常青树,要么是一块还没有归属的墓地. ...

  7. BZOJ 1818 内部白点(离散化+树状数组)

    此题就是1227 的弱化版. 画个图或者稍微证明一下就能够知道,一定不会超过一次变换. 那么我们只需要统计有多少个白点会变黑,换句话说就是有多少个白点上下左右都有黑点. 离散化横坐标,因为没有黑点在的 ...

  8. CodeForces 540E - Infinite Inversions(离散化+树状数组)

    花了近5个小时,改的乱七八糟,终于A了. 一个无限数列,1,2,3,4,...,n....,给n个数对<i,j>把数列的i,j两个元素做交换.求交换后数列的逆序对数. 很容易想到离散化+树 ...

  9. Ultra-QuickSort(归并排序+离散化树状数组)

    Ultra-QuickSort Time Limit: 7000MS   Memory Limit: 65536K Total Submissions: 50517   Accepted: 18534 ...

随机推荐

  1. Caliburn.Micro 杰的入门教程1(翻译)

    Caliburn.Micro 杰的入门教程1(原创翻译)Caliburn.Micro 杰的入门教程2 ,了解Data Binding 和 Events(翻译)Caliburn.Micro 杰的入门教程 ...

  2. 5、Java并发编程:Lock

    Java并发编程:Lock 在上一篇文章中我们讲到了如何使用关键字synchronized来实现同步访问.本文我们继续来探讨这个问题,从Java 5之后,在java.util.concurrent.l ...

  3. 如何设置虚拟化的centos内、外网络通畅

    首先要去确定你的本机(本地物理机)是通过以太网(插网线)上网的,还是通过wifi上网的.这个很重要. 如果是通过以太网去上网,那么虚拟化出来的系统,网络配置应当选择桥接模式. 当然了,也不一定非要用桥 ...

  4. JS原型链与继承别再被问倒了

    原文:详解JS原型链与继承 摘自JavaScript高级程序设计: 继承是OO语言中的一个最为人津津乐道的概念.许多OO语言都支持两种继承方式: 接口继承 和 实现继承 .接口继承只继承方法签名,而实 ...

  5. 181. Flip Bits【LintCode, by java】

    Description Determine the number of bits required to flip if you want to convert integer n to intege ...

  6. 【zabbix 监控】第二章 安装测试被监控主机

    客户端安装测试 一.准备两台被监控主机,分别做如下操作: web129:192.168.19.129 web130:192.168.19.130 [root@web129 ~]#yum -y inst ...

  7. 卸载CDH5.7

    CDH5.7卸载1.记录用户数据目录2.关闭所有服务2.1在CM中,选择某个集群,然后停止集群.2.2逐个关闭CDH中的服务3.删除parcels4.删除集群5.卸载Cloudera manager ...

  8. [git] Git in Practice

    Work flow with git and github Work with Remotes Check the current status git status Check the latest ...

  9. Python中的Numeric

    整型Integer 在Python2.X中,Integer有两种类型,一种是32bit的普通类型,一种是精度无限制的long类型,在数字后面标识l或者L来标识long类型,并且,当32bit发生ove ...

  10. Java学习个人备忘录之数组工具类

    下面主要讲解一个针对数组操作的工具类. a.java -- 工具类文件 //按理来说要先编译本文件, 然后再编译主函数 class ArrayTool { /* 获取整型数组的最大值 */ publi ...