题目大意

  公墓可以看成一块N×M的矩形,矩形的每个格点,要么种着一棵常青树,要么是一块还没有归属的墓地。一块墓地的虔诚度是指以这块墓地为中心的十字架的数目,一个十字架可以看成中间是墓地,墓地的正上、正下、正左、正右都有恰好k棵常青树。小W希望知道他所管理的这片公墓中所有墓地的虔诚度总和是多少。1 ≤ N, M ≤ 1,000,000,000,0 ≤ xi ≤ N,0 ≤ yi ≤ M,1 ≤ W ≤ 100,000,1 ≤ k ≤ 10。

题解

  首先,由N、M的巨大数量和相对地W的较小数量让我们想到离散化。一个点若其所在排和列都没有常青树,则这些点都是无效的。所以我们可以通过离散化将原矩形中的无效部分删去得到一个小矩形。

  然而离散化后我们不可以枚举坐标,这样时间复杂度至少是O(W^2)。所以我们就要以两个点之间的间隔上做文章。

  对于一个点$i$,定义它们上下左右方的常青树(包括它自己)个数分别为$u_i,b_i,l_i,r_i$,它对答案的贡献为$C_{u_i}^k C_{b_i}^k C_{l_i}^k C_{r_i}^k$。对于一排上相邻的两个点$a,b$,它们之间的间隔上每一个点的$C_{l_i}^k C_{r_i}^k$都相等,它们对答案的贡献是$C_{l_a}^k C_{r_a}^k \sum_{i在a,b之间}C_{u_i}^k C_{b_i}^k$。看见sigma了,是不是可以想到树状数组?当处理当前排时,树状数组的key值为列数(也就是架在这一行上),维护的是该行上每一个点的$C_{u_i}^k C_{b_i}^k$的前缀和。这样一行一行地按照列数从小到大(排一下序)扫描常青树,边查询边修改树状数组,这道题就完成了。

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; const int MAX_NODE = 100010, MAX_CANDIDATE_VAL = MAX_NODE * 2, MAX_K = 15;
const unsigned int P = 2147483648u;
int TotRow, TotCol, TotNode, K;
int RowNodeCnt[MAX_CANDIDATE_VAL], ColNodeCnt[MAX_CANDIDATE_VAL];
unsigned int C[MAX_NODE][MAX_K]; struct Discretion
{
private:
int OrgData[MAX_CANDIDATE_VAL], Rank[MAX_CANDIDATE_VAL];
int N; int LowerBound(int l, int r, int k)
{
while (l < r)
{
int mid = (l + r) / 2;
if (k <= OrgData[mid])
r = mid;
else
l = mid + 1;
}
return l;
} public:
Discretion(int n, int *orgData)
{
N = n;
for (int i = 1; i <= N; i++)
OrgData[i] = orgData[i];
sort(OrgData + 1, OrgData + N + 1);
OrgData[0] = -1;
int curRank = 0;
for (int i = 1; i <= N; i++)
Rank[i] = OrgData[i] == OrgData[i - 1] ? curRank : ++curRank;
} int GetRank(int val)
{
return Rank[LowerBound(1, N, val)];
} int GetMaxRank()
{
return Rank[N];
}
}; struct Node
{
int Row, Col; bool operator < (const Node& a) const
{
return Row != a.Row ? Row < a.Row : Col < a.Col;
}
}_nodes[MAX_NODE]; struct BIT
{
private:
unsigned int C[MAX_CANDIDATE_VAL];
int N; int Lowbit(int x)
{
return x & -x;
} public:
BIT(int n):N(n){} void Update(int p, unsigned int delta, bool add)
{
while (p <= N)
{
add ? C[p] += delta : C[p] -= delta;
p += Lowbit(p);
}
} unsigned int Query(int p)
{
unsigned int ans = 0;
while (p >= 1)
{
ans += C[p];
p -= Lowbit(p);
}
return ans;
}
}; void Read()
{
scanf("%d%d%d", &TotRow, &TotCol, &TotNode);
for (int i = 1; i <= TotNode; i++)
scanf("%d%d", &_nodes[i].Row, &_nodes[i].Col);
scanf("%d", &K);
} void GetC()
{
for (int i = 0; i <= TotNode; i++)
{
C[i][0] = 1;
if (i <= K)
C[i][i] = 1;
for (int j = 1; j <= min(i - 1, K); j++)
C[i][j] = C[i - 1][j - 1] + C[i - 1][j];
}
} void Discrete()
{
static int a[MAX_CANDIDATE_VAL];
for (int i = 1; i <= TotNode; i++)
{
a[i * 2 - 1] = _nodes[i].Row;
a[i * 2] = _nodes[i].Col;
}
static Discretion g(TotNode * 2, a);
TotRow = TotCol = g.GetMaxRank();
for (int i = 1; i <= TotNode; i++)
{
_nodes[i].Row = g.GetRank(_nodes[i].Row);
_nodes[i].Col = g.GetRank(_nodes[i].Col);
RowNodeCnt[_nodes[i].Row]++;
ColNodeCnt[_nodes[i].Col]++;
}
} int GetAns()
{
static BIT g(TotCol);
unsigned int ans = 0;
sort(_nodes + 1, _nodes + TotNode + 1);
int pass = 0;//passNodeCntInCurRow
static int colPassCnt[MAX_CANDIDATE_VAL];
for (int i = 1; i < TotNode; i++)
{
pass = _nodes[i - 1].Row == _nodes[i].Row ? pass + 1 : 1;
if (_nodes[i + 1].Row == _nodes[i].Row)
ans += C[pass][K] * C[RowNodeCnt[_nodes[i].Row] - pass][K] * (g.Query(_nodes[i + 1].Col - 1) - g.Query(_nodes[i].Col));
g.Update(_nodes[i].Col, g.Query(_nodes[i].Col) - g.Query(_nodes[i].Col - 1), false);
colPassCnt[_nodes[i].Col]++;
g.Update(_nodes[i].Col, C[colPassCnt[_nodes[i].Col]][K] * C[ColNodeCnt[_nodes[i].Col] - colPassCnt[_nodes[i].Col]][K], true);
}
return (int)(ans % P);
} int main()
{
Read();
GetC();
Discrete();
printf("%d\n", GetAns());
return 0;
}

  

luogu2154 [SDOI2009] 虔诚的墓主人 离散化 树状数组 扫描线的更多相关文章

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

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

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

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

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

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

  4. BZOJ 1227 虔诚的墓主人(离散化+树状数组)

    题目中矩形的尺寸太大,导致墓地的数目太多,如果我们统计每一个墓地的虔诚度,超时是一定的. 而常青树的数目<=1e5.这启发我们从树的方向去思考. 考虑一行没有树的情况,显然这一行的墓地的虔诚度之 ...

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

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

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

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

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

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

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

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

  9. HDU 5862 Counting Intersections(离散化+树状数组)

    HDU 5862 Counting Intersections(离散化+树状数组) 题目链接http://acm.split.hdu.edu.cn/showproblem.php?pid=5862 D ...

随机推荐

  1. DeltaFish 小组成员及个人博客地址

    艾寅中  http://www.cnblogs.com/aiyz 陈志锴  http://www.cnblogs.com/chenzhikai 李   鑫  http://www.cnblogs.co ...

  2. 开发日记(项目中SQL查询的优化)

    今天发现自己之前写的一些SQL查询在执行效率方面非常不理想,于是尝试做了些改进. 需求为查询国地税表和税源表中,国税有而税源没有的条目数,之前的查询如下: SELECT COUNT(NAME)     ...

  3. 移动web——bootstrap如何修改原组件

    基本介绍 1.bootstrap提供了丰富的组件,但是有时候我们不仅要删除不必要的标签,还需要修改里面的样式 2.我们建议若是修改样式那么最好将源样式从css中拷贝出来,名字换掉,然后修改具体样式,这 ...

  4. servlet得到web应用路径

    package context; import java.io.IOException; import javax.servlet.ServletContext; import javax.servl ...

  5. Python标准库sys

    1.命令行参数sys.argv 我们从Python语言之模块第一部分的例子开始,看看sys.argv中到底存了些什么内容. #Filename: using_sys.py import sys i=0 ...

  6. 【转载】HTTP 响应头与状态码

    原文地址:https://segmentfault.com/a/1190000006689786 HTTP Response Header 响应头域允许服务器传递不能放在状态行的附加信息,这些域主要描 ...

  7. DAMA

    无论是小数据时代还是大数据时代,数据治理都是个非常重要的工作,数据质量问题是个非常普遍的问题.对于传统企业来说,核心业务还是流程驱动的,需要而且有条件把数据做准确,这就需要在数据管理上面下功夫. 介绍 ...

  8. HDU_1063_Exponentiation_大数

    Exponentiation Time Limit: 2000/500 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Tot ...

  9. Django的文件下载

    在实际的项目中很多时候需要用到下载功能,如导excel.pdf或者文件下载,当然你可以使用web服务自己搭建可以用于下载的资源服务器,如nginx,这里我们主要介绍django中的文件下载. 我们这里 ...

  10. Unity中确定时间是否在一定范围内

    NowTime = DateTime.Now.ToLocalTime(); Timeyear = DateTime.Now.ToLocalTime().ToString("yyyy-MM-d ...