【LG2154】[SDOI2009]虔诚的墓主人

题面

洛谷

题解

如果您没有看懂题,请反复阅读题面及样例

可以发现,对于某一个点,它的答案就是上下左右几个组合数乘起来。

这样直接做复杂度显然爆炸,考虑怎么优化这个东西。

我们可以固定左右,在某两个左右之间维护上下有多少个,这样子的话左右的贡献就是不变的,而且你最多只会变化\(O(n)\)次左右边界,复杂度有保证。

这样的话,每次查询一个左右边界内上下的贡献,用线段树维护即可。(描述可能有些模糊,具体详见代码)

代码

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <vector>
using namespace std;
inline int gi() {
register int data = 0, w = 1;
register char ch = 0;
while (!isdigit(ch) && ch != '-') ch = getchar();
if (ch == '-') w = -1, ch = getchar();
while (isdigit(ch)) data = 10 * data + ch - '0', ch = getchar();
return w * data;
}
const int MAX_N = 1e5 + 5;
#define lson (o << 1)
#define rson (o << 1 | 1)
int sum[MAX_N << 2];
void modify(int o, int l, int r, int pos, int v) {
if (l == r) return (void)(sum[o] = v);
int mid = (l + r) >> 1;
if (pos <= mid) modify(lson, l, mid, pos, v);
else modify(rson, mid + 1, r, pos, v);
sum[o] = sum[lson] + sum[rson];
}
int query(int o, int l, int r, int ql, int qr) {
if (ql <= l && r <= qr) return sum[o];
int mid = (l + r) >> 1, res = 0;
if (ql <= mid) res += query(lson, l, mid, ql, qr);
if (qr > mid) res += query(rson, mid + 1, r, ql, qr);
return res;
}
int W, H, N, K;
int x[MAX_N], y[MAX_N], ox[MAX_N], oy[MAX_N], tx, ty;
vector<int> vec[MAX_N];
int C[MAX_N][15], c1[MAX_N], c2[MAX_N];
bool cmp(const int &l, const int &r) { return y[l] < y[r]; }
int main () {
#ifndef ONLINE_JUDGE
freopen("cpp.in", "r", stdin);
#endif
W = gi(), H = gi(), N = gi();
for (int i = 1; i <= N; i++) x[i] = ox[i] = gi(), y[i] = oy[i] = gi();
K = gi();
sort(&ox[1], &ox[N + 1]), sort(&oy[1], &oy[N + 1]);
tx = unique(&ox[1], &ox[N + 1]) - ox - 1;
ty = unique(&oy[1], &oy[N + 1]) - oy - 1;
for (int i = 1; i <= N; i++) {
x[i] = lower_bound(&ox[1], &ox[tx + 1], x[i]) - ox;
y[i] = lower_bound(&oy[1], &oy[ty + 1], y[i]) - oy;
}
for (int i = 1; i <= N; i++) vec[x[i]].push_back(i), c1[y[i]]++;
for (int i = 1; i <= tx; i++) vec[i].push_back(N + 1);
y[N + 1] = ty + 1;
for (int i = 1; i <= tx; i++) sort(vec[i].begin(), vec[i].end(), cmp);
for (int i = 0; i <= N; i++) C[i][0] = 1;
for (int i = 1; i <= N; i++)
for (int j = 1; j <= min(i, K); j++)
C[i][j] = C[i - 1][j] + C[i - 1][j - 1];
int ans = 0;
for (int i = 1; i <= tx; i++) {
int res = 0, sz = vec[i].size();
for (int j = 0; j < sz; j++) {
int l = j ? y[vec[i][j - 1]] : 0, r = y[vec[i][j]];
if (l + 1 <= r - 1) ans += res * query(1, 1, ty, l + 1, r - 1);
if (j == sz - 1) break;
res = C[j + 1][K] * C[sz - j - 2][K];
++c2[r];
modify(1, 1, ty, r, C[c2[r]][K] * C[c1[r] - c2[r]][K]);
}
}
if (ans < 0) ans += 2147483648ll;
printf("%d\n", ans);
return 0;
}

【LG2154】[SDOI2009]虔诚的墓主人的更多相关文章

  1. BZOJ 1227: [SDOI2009]虔诚的墓主人

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

  2. Bzoj 1227: [SDOI2009]虔诚的墓主人 树状数组,离散化,组合数学

    1227: [SDOI2009]虔诚的墓主人 Time Limit: 5 Sec  Memory Limit: 259 MBSubmit: 895  Solved: 422[Submit][Statu ...

  3. bzoj1227 [SDOI2009]虔诚的墓主人(组合公式+离散化+线段树)

    1227: [SDOI2009]虔诚的墓主人 Time Limit: 5 Sec  Memory Limit: 259 MBSubmit: 803  Solved: 372[Submit][Statu ...

  4. 1227: [SDOI2009]虔诚的墓主人

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

  5. bzoj1227 P2154 [SDOI2009]虔诚的墓主人

    P2154 [SDOI2009]虔诚的墓主人 组合数学+离散化+树状数组 先看题,结合样例分析,易得每个墓地的虔诚度=C(正左几棵,k)*C(正右几棵,k)*C(正上几棵,k)*C(正下几棵,k),如 ...

  6. 【BZOJ1227】[SDOI2009]虔诚的墓主人(线段树)

    [BZOJ1227][SDOI2009]虔诚的墓主人(线段树) 题面 BZOJ 洛谷 题解 显然发现答案就是对于每一个空位置,考虑上下左右各有多少棵树,然后就是这四个方向上树的数量中选\(K\)棵出来 ...

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

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

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

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

  9. luoguP2154 [SDOI2009]虔诚的墓主人

    SDOI2009虔诚的墓主人 喜闻乐见,我终于把此题读懂了..所以可以写了. 其实就是让我们求有多少个十字架 一个十字架的定义为中间有一个空地 周围4个正方向都有k棵树. 不难想到nm的暴力 我们预处 ...

随机推荐

  1. WPF矢量字体图标(iconfont)

    原文:WPF矢量字体图标(iconfont) 转载:点击打开链接 步骤: 一.下载添加iconfont文件 二.添加到资源文件夹,并设置不复制,且为资源文件 三.增加FIcon.xaml文件 < ...

  2. 8、VUE自定义组件

    1.为什么要使用自定义组件? 自定义组件是用来封装复杂的内容,提高可重用性,比如封装复杂的表格组件.日历组件.图片轮播组件等. 2.自定义组件 2.1. 全局组件 全局组件是每个Vue对象都能使用的组 ...

  3. 深入理解TCP/IP传输层

    传输层:负责数据能够从发送端传到接收端(只需要关注点对点的传输,中间的传输过程一概不管) UDP和TCP UDP(全双工):1.无连接,2不可靠,3.面向数据报 分别表示UDP源端口号.目的端口号.U ...

  4. rsyslog详解实战和避坑

    目标是要把线上环境的debug日志及集中化收集起来,一方面是方便开发调试:一方面是避免直接到线上环境查看,存在安全隐患. 常用可选方案: rsyslog发送端 + rsyslog接收端: 直接存在接收 ...

  5. Unity PhysicsScene测试

    应该是unity 2018.3中加入的功能,对象可以放置于不同的物理场景中. 一个Scene对应一个物理场景(PhysicsScene),若想放入独立的物理场景测试创建一个Scene即可.见下图gif ...

  6. python的new与init

    基于文章:Why is init() always called after new()? 特别说明: 这篇文章的灵感来源于stackoverflow的一个提问,准确说,我只是做了些知识梳理的工作,对 ...

  7. IntelliJ Cannot find declaration to goto----解决方案

    系统中已经有了该类库,还是找不到类提示 close the project in intellij. close intellij. go to the project folder and dele ...

  8. Beego 学习笔记14:Session控制

    Session控制 1>     Session常用来作为全局变量使用,比如记录当前登录的用户,或者页面之间传递数据使用. 2>     Beego框架内置了 session 模块,目前 ...

  9. 41、css总结

      1.阴影:box-shadow:0 5px 20px rgba(0,0,0,.1); 2.css实现滚动进度条效果: body { position: relative; padding: 50p ...

  10. 面试题:什么叫B*树

    B*-tree是B+-tree的变体,在B+树的基础上(所有的叶子结点中包含了全部关键字的信息,及指向含有这些关键字记录的指针),B*树中非根和非叶子结点再增加指向兄弟的指针: