题目传送门

  传送点I

  传送点II

题目大意

  平面上有$n$个点。问存在多少个矩形使得只有左下角和右上角有点。

  考虑枚举左下角这个点。然后看一下是个什么情况:

  嗯对,是个单调栈。但不可能暴力去求每个点右侧和上方的点的单调栈。

  注意到如果给单调栈设个上界,那么顶多会削掉一些点,不会发生大的改变。

  考虑CDQ分治,然后按照$y$从大到小排序。枚举左边的点然后不断把右边纵坐标大于它的点加入单调栈。(把横坐标比它大的全弹掉)

  然后还需要考虑一个问题:

  

  绿色点上方的点不能选。

  如何找到这个上界?对左边开一个单调栈。然后在单调栈上二分。然后在右边的单调栈上二分可以找到有多少点不能选。

  (今天考这道题,怀疑自己是zz,左边写一个线段树找上界,右边二分了两次)

Code

 /**
* bzoj
* Problem#4237
* Accepted
* Time: 7540ms
* Memory: 8332k
*/
#include <algorithm>
#include <iostream>
#include <cstdlib>
#include <cstdio>
#include <ctime>
#ifndef WIN32
#define Auto "%lld"
#else
#define Auto "%I64d"
#endif
using namespace std;
typedef bool boolean;
#define ll long long
const signed int inf = (signed) (~0u >> ); typedef class IO {
public:
int sta; }IO; int __tp = ;
char __buf[]; #define digit(_x) ((_x) >= '0' && (_x) <= '9')
#define pick(_x) ((_x) = getchar()) IO& operator >> (IO& io, int& u) {
char x;
while (~(x = getchar()) && !digit(x));
for (u = x - ''; x = getchar(), digit(x); u = u * + x - '');
return io;
} IO in; template<typename T>
void alloc(T*& p, int siz) {
p = new T[(siz + )];
} typedef class Point {
public:
int x, y; boolean operator < (Point b) const {
if (y ^ b.y) return y > b.y;
return x < b.x;
}
}Point; int n;
ll res = ;
int *buf;
Point *ps, *bp;
Point *pt, *vt; inline void init() {
in >> n;
alloc(vt, n);
alloc(ps, n), alloc(bp, n);
alloc(buf, n), alloc(pt, n);
for (int i = ; i <= n; i++)
in >> ps[i].x >> ps[i].y;
} inline void discrete() {
for (int i = ; i <= n; i++)
buf[i] = ps[i].x;
sort(buf + , buf + n + );
for (int i = ; i <= n; i++)
ps[i].x = lower_bound(buf + , buf + n + , ps[i].x) - buf;
for (int i = ; i <= n; i++)
buf[i] = ps[i].y;
sort(buf + , buf + n + );
for (int i = ; i <= n; i++)
ps[i].y = lower_bound(buf + , buf + n + , ps[i].y) - buf;
} int search(Point* pt, int tp, int y) {
int l = , r = tp;
while (l <= r) {
int mid = (l + r) >> ;
if (pt[mid].y <= y)
r = mid - ;
else
l = mid + ;
}
return r + ;
} void dividing(int l, int r, int ql, int qr) {
if (l == r || ql > qr)
return ;
int mid = (l + r) >> ;
int pl = l - , pr = r + ;
for (int i = ql; i <= qr; i++)
if (ps[i].x <= mid)
bp[++pl] = ps[i];
else
bp[--pr] = ps[i];
reverse(bp + pr, bp + qr + );
for (int i = ql; i <= qr; i++)
ps[i] = bp[i]; int tp = , tp1 = ;
vt[] = (Point) {inf, inf};
for (int i = l; i <= pl; i++) {
while (pr <= r && ps[pr].y > ps[i].y) {
while (tp && pt[tp].x >= ps[pr].x)
tp--;
pt[++tp] = ps[pr], pr++;
}
int l = , r = tp1;
while (l <= r) {
int mid = (l + r) >> ;
if (vt[mid].x >= ps[i].x)
l = mid + ;
else
r = mid - ;
}
res += tp - search(pt, tp, vt[l - ].y) + ;
while (tp1 && vt[tp1].x <= ps[i].x)
tp1--;
vt[++tp1] = ps[i];
} dividing(l, mid, ql, pl);
dividing(mid + , r, pl + , qr);
} inline void solve() {
sort(ps + , ps + n + );
dividing(, n, , n);
printf(Auto, res);
} int main() {
init();
discrete();
solve();
return ;
}

bzoj 4237 稻草人 - CDQ分治 - 单调栈的更多相关文章

  1. 【BZOJ4237】 稻草人 CDQ分治+单调栈

    ## 题目描述 JOI村有一片荒地,上面竖着N个稻草人,村民们每年多次在稻草人们的周围举行祭典. 有一次,JOI村的村长听到了稻草人们的启示,计划在荒地中开垦一片田地.和启示中的一样,田地需要满足以下 ...

  2. bzoj 4237: 稻草人 -- CDQ分治

    4237: 稻草人 Time Limit: 40 Sec  Memory Limit: 256 MB Description JOI村有一片荒地,上面竖着N个稻草人,村民们每年多次在稻草人们的周围举行 ...

  3. bzoj4237: 稻草人 cdq分治 单调栈

    目录 题目链接 题解 代码 题目链接 bzoj4237: 稻草人 题解 暴力统计是n^2的 考虑统计一段区间对另一端的贡献 对于y值cdq分治,降调一维 对于当前两个分治区间统计上面那部分对下面那部分 ...

  4. 【BZOJ4237】稻草人 cdq分治+单调栈+二分

    [BZOJ4237]稻草人 Description JOI村有一片荒地,上面竖着N个稻草人,村民们每年多次在稻草人们的周围举行祭典. 有一次,JOI村的村长听到了稻草人们的启示,计划在荒地中开垦一片田 ...

  5. [BZOJ4237]稻草人:CDQ分治+单调栈

    分析 按\(y\)排序后CDQ分治,可以发现每个点可以影响的是\(x\)坐标的一段区间,可以使用扫描线+单调栈,在单调栈上二分即可解决,时间复杂度\(O(n \log^2 n)\). 通过归并排序可以 ...

  6. Loj#2880-「JOISC 2014 Day3」稻草人【CDQ分治,单调栈,二分】

    正题 题目链接:https://loj.ac/problem/2880 题目大意 给出平面上的\(n\)个点,然后求有多少个矩形满足 左下角和右上角各有一个点 矩形之间没有其他点 \(1\leq n\ ...

  7. bzoj 4237 稻草人 CDQ

    稻草人 Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 1433  Solved: 626[Submit][Status][Discuss] Descr ...

  8. 【bzoj4237】稻草人 分治+单调栈+二分

    题目描述 JOI村有一片荒地,上面竖着N个稻草人,村民们每年多次在稻草人们的周围举行祭典. 有一次,JOI村的村长听到了稻草人们的启示,计划在荒地中开垦一片田地.和启示中的一样,田地需要满足以下条件: ...

  9. [BZOJ 3456]城市规划(cdq分治+FFT)

    [BZOJ 3456]城市规划(cdq分治+FFT) 题面 求有标号n个点无向连通图数目. 分析 设\(f(i)\)表示\(i\)个点组成的无向连通图数量,\(g(i)\)表示\(i\)个点的图的数量 ...

随机推荐

  1. 39.css3----button按钮点击时出现蓝色边框

    css控制Button 按钮的点击时候出现蓝色边框http://www.inbeijing.org/archives/1139 Button 按钮的点击时候出现蓝色边框的问题 添加css属性,这样在点 ...

  2. HDU 2842 Chinese Rings(常数矩阵)

    Chinese Rings 转载自:点这里 [题目链接]Chinese Rings [题目类型]常数矩阵 &题意: 一种中国环,解开第k个环需要先解开全部的前(k-2)个环,并留有第(k-1) ...

  3. 基于bootstrap的jQuery多级列表树插件

    简要教程 bootstrap-treeview是一款效果非常酷的基于bootstrap的jQuery多级列表树插件.该jQuery插件基于Twitter Bootstrap,以简单和优雅的方式来显示一 ...

  4. 水题C

    某校大门外长度为L的马路上有一排树,每两棵相邻的树之间的间隔都是1米.我们可以把马路看成一个数轴,马路的一端在数轴0的位置,另一端在L的位置:数轴上的每个整数点,即0,1,2,……,L,都种有一棵树. ...

  5. git server 配置

    因为后面要采用Git代替Subversion,花了点时间配置了Git服务端和客户端,像以前一样,仍然基于最新的Ubuntu11.10 server/desktop系统. 感谢这几篇文章的作者: htt ...

  6. html5-列表

    <!DOCTYPE html><html lang="en"><head>    <meta charset="UTF-8&qu ...

  7. Python 3 -- 数据结构(list, dict, set,tuple )

    看了<Head First Python>后,觉得写的很不错,适合新手.此处为读书笔记,方便日后查看. Python 提供了4中数据结构:list,dict,set,tuple. 每种结构 ...

  8. python 数据序列化(json、pickle、shelve)

    本来要查一下json系列化自定义对象的一个问题,然后发现这篇博客(https://www.cnblogs.com/yyds/p/6563608.html)很全面,感谢作者,关于python序列化的知识 ...

  9. Linux基础命令---arch

    Arch         Arch指令主要用于显示当前主机的硬件结构类型,我们可以看到它输出的结果有:i386.i486.mips.alpha等.此命令的适用范围:RedHat.RHEL.Ubuntu ...

  10. 转:【专题七】UDP编程补充——UDP广播程序的实现

    上次因为时间的关系,所以把上一个专题遗留下的一个问题在本专题中和大家分享下,本专题主要介绍下如何实现UDP广播的程序,下面就直接介绍实现过程和代码以及运行的结果. 一.程序实现 UDP广播程序的实现代 ...