1227: [SDOI2009]虔诚的墓主人

Time Limit: 5 Sec  Memory Limit: 259 MB
Submit: 1412  Solved: 664
[Submit][Status][Discuss]

Description

小W 是一片新造公墓的管理人。公墓可以看成一块N×M 的矩形,矩形的每个格点,要么种着一棵常青树,要么是一块还没有归属的墓地。当地的居民都是非常虔诚的基督徒,他们愿意提前为自己找一块合适墓地。为了体现自己对主的真诚,他们希望自己的墓地拥有着较高的虔诚度。一块墓地的虔诚度是指以这块墓地为中心的十字架的数目。一个十字架可以看成中间是墓地,墓地的正上、正下、正左、正右都有恰好k 棵常青树。小W 希望知道他所管理的这片公墓中所有墓地的虔诚度总和是多少

Input

第一行包含两个用空格分隔的正整数N 和M,表示公墓的宽和长,因此这个矩形公墓共有(N+1) ×(M+1)个格点,左下角的坐标为(0, 0),右上角的坐标为(N, M)。第二行包含一个正整数W,表示公墓中常青树的个数。第三行起共W 行,每行包含两个用空格分隔的非负整数xi和yi,表示一棵常青树的坐标。输入保证没有两棵常青树拥有相同的坐标。最后一行包含一个正整数k,意义如题目所示。

Output

包含一个非负整数,表示这片公墓中所有墓地的虔诚度总和。为了方便起见,答案对2,147,483,648 取模。

Sample Input

5 6
13
0 2
0 3
1 2
1 3
2 0
2 1
2 4
2 5
2 6
3 2
3 3
4 3
5 2
2

Sample Output

6

HINT

图中,以墓地(2, 2)和(2, 3)为中心的十字架各有3个,即它们的虔诚度均为3。其他墓地的虔诚度为0。

所有数据满足1 ≤ N, M ≤ 1,000,000,000,0 ≤ xi ≤ N,0 ≤ yi ≤ M,1 ≤ W ≤ 100,000, 1 ≤ k ≤ 10。存在50%的数据,满足1 ≤ k ≤ 2。存在25%的数据,满足1 ≤ W ≤ 10000。

注意:”恰好有k颗树“,这里的恰好不是有且只有,而是从>=k的树中恰好选k棵

首先要离散坐标    由于如果一个点正上下左右都没有其他点,那么他是肯定不会贡献答案的,所以只用记录所有点的横纵坐标,重新构成坐标系统计答案

l[]表示它同行左边的点个个数,r[]表示右边,u[]表示同列上面的点的个数,d表示同列下面的点数 c[]表示组合数

对于纵坐标相同且横坐标相邻的两点设为a,b,那么他们中间的一段无点的区间可以贡献的答案就是c[l[a]][k]*c[r[b]][k]*sigma(c[u[i]][k]*c[d[i]][k])  (a<i<b)

对于中间那一段,可以用bit动态统计。bit下标为横坐标,表示在当前列横坐标为x选择的方案数,用num统计当前列横坐标为x正下方的点数方便转移

当新扫描到一个点,这个点就会对bit进行影响,num[x]++,此位置的方案数也会变化

具体要看代码实现

  1. #include<cstdio>
  2. #include<iostream>
  3. #include<algorithm>
  4. #include<cstring>
  5. #include<queue>
  6. #define inf 0x3f3f3f3f
  7. #define ll long long
  8. #define N 1000050
  9. #define mod 2147483648LL
  10. using namespace std;
  11. int n,m,w,k,num[N<<1],h[N<<1],l[N<<1],b[N<<1];
  12. ll v[N<<1],c[N][12];
  13. void update(int x,int val){
  14. while(x<=2*w){
  15. v[x]+=val;v[x]%=mod;
  16. x+=x&-x;
  17. }
  18. }
  19. ll query(int x){
  20. ll res=0;
  21. while(x){
  22. res+=v[x];res%=mod;
  23. x-=x&-x;
  24. }
  25. return res;
  26. }
  27. struct tree{
  28. int x,y;
  29. bool operator < (const tree &b)const{
  30. return y==b.y?x<b.x:y<b.y;
  31. }
  32. }a[N];
  33. void pre(){
  34. for(int i=0;i<=w;i++)c[i][0]=c[i][i]=1;
  35. for(int i=1;i<=w;i++)
  36. for(int j=1;j<=min(i,k);j++)
  37. c[i][j]=(c[i-1][j-1]+c[i-1][j])%mod;
  38. }
  39. int main(){
  40. scanf("%d%d%d",&n,&m,&w);
  41. for(int i=1;i<=w;i++){
  42. scanf("%d%d",&a[i].x,&a[i].y);
  43. b[i]=a[i].x;b[i+w]=a[i].y;
  44. }scanf("%d",&k);pre();
  45.  
  46. sort(b+1,b+1+w*2);sort(a+1,a+1+w);
  47. int len=unique(b+1,b+1+w*2)-b-1;
  48. for(int i=1;i<=w;i++){
  49. int px=lower_bound(b+1,b+len+1,a[i].x)-b;
  50. int py=lower_bound(b+1,b+len+1,a[i].y)-b;
  51. h[py]++;l[px]++;
  52. }
  53. int cnt=0;ll ans=0;
  54. for(int i=1;i<=w;i++){
  55. int px=lower_bound(b+1,b+len+1,a[i].x)-b;
  56. int py=lower_bound(b+1,b+len+1,a[i].y)-b;
  57. int pre=lower_bound(b+1,b+len+1,a[i-1].x)-b;
  58. if(i>1&&a[i].y==a[i-1].y){
  59. cnt++;
  60. ll t1=c[cnt][k]*c[h[py]-cnt][k];
  61. ll t2=query(px-1)-query(pre);
  62. ans+=t1*t2;ans%=mod;
  63. }
  64. else cnt=0;
  65. num[px]++;int t=num[px];
  66. int change=(c[t][k]*c[l[px]-t][k]-c[t-1][k]*c[l[px]-t+1][k])%mod;
  67. update(px,change);
  68. //printf("%lld %lld\n",change,ans);
  69. }
  70. if(ans<0)ans+=mod;cout<<ans;
  71. return 0;
  72. }

bzoj1227 组合数学+bit的更多相关文章

  1. BZOJ-1227 虔诚的墓主人 树状数组+离散化+组合数学

    1227: [SDOI2009]虔诚的墓主人 Time Limit: 5 Sec Memory Limit: 259 MB Submit: 914 Solved: 431 [Submit][Statu ...

  2. 【bzoj1227】 SDOI2009—虔诚的墓主人

    http://www.lydsy.com/JudgeOnline/problem.php?id=1227 (题目链接) 题意 一个n*m的公墓,一个点上要么是墓地,要么是常青树,给出一个数K,并规定每 ...

  3. poj 3734 Blocks 快速幂+费马小定理+组合数学

    题目链接 题意:有一排砖,可以染红蓝绿黄四种不同的颜色,要求红和绿两种颜色砖的个数都是偶数,问一共有多少种方案,结果对10007取余. 题解:刚看这道题第一感觉是组合数学,正向推了一会还没等推出来队友 ...

  4. 组合数学or not ---- n选k有重

    模板问题: 1. 取物品 (comb.pas/c/cpp) [问题描述] 现在有n个物品(有可能相同),请您编程计算从中取k个有多少种不同的取法.[输入] 输入文件有两行,第一行包含两个整数n,k(2 ...

  5. 组合数学(全排列)+DFS CSU 1563 Lexicography

    题目传送门 /* 题意:求第K个全排列 组合数学:首先,使用next_permutation 函数会超时,思路应该转变, 摘抄网上的解法如下: 假设第一位是a,不论a是什么数,axxxxxxxx一共有 ...

  6. uestc1888 Birthday Party    组合数学,乘法原理

    题目链接:http://acm.hust.edu.cn/vjudge/contest/view.action?cid=25539#problem/G 题目意思: 有n个人,每个人有一个礼物,每个人能拿 ...

  7. UVA 11076 Add Again 计算对答案的贡献+组合数学

    A pair of numbers has a unique LCM but a single number can be the LCM of more than one possiblepairs ...

  8. POJ3252——Round Number(组合数学)

    Round Numbers DescriptionThe cows, as you know, have no fingers or thumbs and thus are unable to pla ...

  9. HDU4675【GCD of scequence】【组合数学、费马小定理、取模】

    看题解一开始还有地方不理解,果然是我的组合数学思维比较差 然后理解了之后自己敲了一个果断TLE.... 我以后果然还得多练啊 好巧妙的思路啊 知识1: 对于除法取模还需要用到费马小定理: a ^ (p ...

随机推荐

  1. 【技巧】Java工程中的Debug信息分级输出接口

    也许本文的标题你们没咋看懂.但是,本文将带大家领略输出调试的威力. 灵感来源 说到灵感,其实是源于笔者在修复服务器的ssh故障时的一个发现. 这个学期初,同袍(容我来一波广告产品页面,同袍官网)原服务 ...

  2. redux的知识点

    Redux: Redux 是针对 JavaScript应用的可预测状态容器 就是用来管理数据的.stroe 保存数据action领导 下达命令reducer员工 执行命令 下载命令:  npm ins ...

  3. CSS你所不知的伪元素的用法

    你所不知的 CSS ::before 和 ::after 伪元素用法 博客分类: Div / Css / XML / HTML5   CSS 有两个说不上常用的伪类 :before 和 :after, ...

  4. php后台的在控制器中就可以实现阅读数增加

    $smodel=M('Sswz');$smodel->where($map)->setInc('view' ,1);php后台的在控制器中就可以实现阅读数增加前台不需要传值

  5. Spark学习笔记之RDD中的Transformation和Action函数

    总算可以开始写第一篇技术博客了,就从学习Spark开始吧.之前阅读了很多关于Spark的文章,对Spark的工作机制及编程模型有了一定了解,下面把Spark中对RDD的常用操作函数做一下总结,以pys ...

  6. Linq GroupJoin

    static void Main(string[] args) { List<Person> persons = new List<Person> { }, }, }; Lis ...

  7. python利用文件对话框获取文件路径

    一.单文件 python3: import tkinter as tk from tkinter import filedialog root = tk.Tk() root.withdraw() fi ...

  8. servlet2.3/2.5/3.0/3.1的xml名称空间备忘

    The web.xml is a configuration file to describe how a web application should be deployed. Here’re 5  ...

  9. 简单搭建SpringMVC框架详解

    在公司待了两年,用的一直是Spring+SpringMVC+Hibernate框架,都是公司自己搭建好的,自己从来没有主动搭建过,闲来无聊,自己搭建试试.一下即我搭建的过程以及搭建所遇到的问题,有部分 ...

  10. re模块中的compile函数

    compile compile(pattern,flag=0) compile a regular expression pattern,return a pattern object compile ...