Luogu P5490

作为一道模板题让我卡了一个月……

对于线段树+离散化新手而言这实在是太难了……

有关离散化:

可以查看这一篇文章:https://www.jianshu.com/p/9347659dcf18
一种缩小数据范围的小技巧,在本题中用于缩短线段树的区间大小

基本思路

想象一条直线,从左往右扫,我们要计算的就是直线扫过的面积。

很容易可以想出把整个图形按照每一条竖线切割,分割成几个矩形。

具体做法可以看代码。

#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
long long tag[1600005],tree[1600005],cnt,ans,n,x,x2,y,y2,line[1600005];
struct data
{
long long x,l,r,flag;
}edge[1600005];
bool comp(data a,data b)
{
if (a.x!=b.x)return a.x<b.x;
else return a.l<b.l;
}
/*void push_down(int root,int l,int r)
{
int lson=root<<1;
int rson=root<<1|1;
int mid=(l+r)>>1;
tag[lson]+=tag[root];
if (tag[lson]) tree[lson]=line[mid]-line[l];
tag[rson]+=tag[root];
if (tag[rson]) tree[rson]=line[r]-line[mid];
tag[root]=0;
}*/
void add(int root,int l,int r,long long al,long long ar,int num)
{
if (al<=line[l]&&line[r]<=ar)
{
tag[root]+=num;
if (!tag[root]) tree[root]=tree[root<<1]+tree[root<<1|1];
if (tag[root]) tree[root]=line[r]-line[l];
return ;
//tag存储节点代表的区间被覆盖的次数
//tree存储节点代表的区间被覆盖的长度
//值得注意的是,即使当前区间没有被完全覆盖,tree[root]也不一定为0。
}
if (l+1==r) return ;
if (ar<=line[l]||al>=line[r]) return ;//加入等号是因为若只有端点相同,实质上也是没有交集的。
//push_down(root,l,r);
//无需pushdown的原因为:对于任意一个区间,完全覆盖它的竖线一定是成对的(因为有左右两边),即有进有出,不可能会出现负值。
//并且对于任意区间的子区间,子区间是否被覆盖或者被覆盖的长度对父区间毫无影响
//如果加入pushdown,会导致超时。
int mid=(l+r)>>1;
if (al<line[mid]) add(root<<1,l,mid,al,ar,num);
if (ar>line[mid]) add(root<<1|1,mid,r,al,ar,num);
if (tag[root]==0) tree[root]=tree[root<<1]+tree[root<<1|1];
else tree[root]=line[r]-line[l]; }
int main()
{
scanf("%lld",&n);
for (int i=1;i<=n;i++)
{
scanf("%lld%lld%lld%lld",&x,&y,&x2,&y2);
edge[++cnt].x=x;
edge[cnt].l=y;
edge[cnt].r=y2;
edge[cnt].flag=1;
line[cnt]=y;
edge[++cnt].x=x2;
edge[cnt].l=y;
edge[cnt].r=y2;
edge[cnt].flag=-1;
line[cnt]=y2;
//edge记录竖线的位置
//line记录每一个点的位置,方便离散化处理
}
sort(edge+1,edge+1+cnt,comp);
sort(line+1,line+1+cnt);
int newend=unique(line+1,line+1+cnt)-line-1;//去重函数,此时newend恰好是line最后一个元素的下标
for (int i=1;i<cnt;i++)
{
add(1,1,newend,edge[i].l,edge[i].r,edge[i].flag);
ans+=tree[1]*(edge[i+1].x-edge[i].x);//整个区间被覆盖的总长度*到下一条竖边的距离,得到这一部分的面积
//最后一条竖边不需要处理
}
printf("%lld",ans);
return 0;
}

【Luogu P5490】扫描线的更多相关文章

  1. Luogu P5490 扫描线

    模板题,想象一条线从左边扫到右边,只有在矩阵边界才会产生影响,所以我们离散化缩小数据范围,再用线段树维护扫描线上的情况,得出结果 #include<bits/stdc++.h> #defi ...

  2. BZOJ 4814 Luogu P3699 [CQOI2017]小Q的草稿 (计算几何、扫描线、set)

    题目链接 (BZOJ) http://lydsy.com/JudgeOnline/problem.php?id=4814 (Luogu) https://www.luogu.org/problem/P ...

  3. luogu P1856 [USACO5.5]矩形周长Picture 扫描线 + 线段树

    Code: #include<bits/stdc++.h> #define maxn 200007 #define inf 100005 using namespace std; void ...

  4. BZOJ 2731 Luogu P3219 [HNOI2012]三角形覆盖问题 (扫描线)

    题目链接: (bzoj)https://www.lydsy.com/JudgeOnline/problem.php?id=2731 (luogu)https://www.luogu.org/probl ...

  5. [P5490] 【模板】扫描线 - 线段树

    求 \(n\) 个矩形的面积并 Solution 将矩形转化为 \(y_1\) 位置的 + 修改 和 \(y_2\) 位置的 - 修改.然后按照 \(+y\) 顺序依次处理所有的修改,到达的一个新的位 ...

  6. [CERC2017]Intrinsic Interval——扫描线+转化思想+线段树

    [CERC2017]Intrinsic Interval https://www.luogu.org/blog/ywycasm/solution-p4747# 这种“好的区间”,见得还是比较多的了. ...

  7. 【学习笔记】线段树—扫描线补充 (IC_QQQ)

    [学习笔记]线段树-扫描线补充 (IC_QQQ) (感谢 \(IC\)_\(QQQ\) 大佬授以本内容的著作权.此人超然于世外,仅有 \(Luogu\) 账号 尚可膜拜) [学习笔记]线段树详解(全) ...

  8. 【Luogu P1502】窗口的星星

    Luogu P1502 题意很好理解,就是问给出的矩形套住的最大和. 但是做起来却十分麻烦. --来自疯狂爆10分的愤怒 一个比较高效的思路是--把每一个星星作为左下角向右上方拓展形成一个矩形, 拓展 ...

  9. luogu P1382 楼房

    二次联通门 : luogu P1382 楼房 /* luogu P1382 楼房 线段树 + 扫描线 + 离散化 正解貌似是堆... MMP...二段式线段树各种错误... 离散化一下横坐标 扫描线扫 ...

随机推荐

  1. 别说家庭组开不了,一定成功的方法|win7家庭组无法开启|win7如何共享打印机

    两台WIN7要开启家庭组并且共享打印机 这两台电脑必须打开以下服务:dnscache(任务管理器 简写).fdrespub(任务管理器 简写).(接下来下了都是服务里面的)SSDP Discovery ...

  2. OptimalSolution(2)--二叉树问题(2)BST、BBT、BSBT

    一.判断二叉树是否为平衡二叉树(时间复杂度O(N)) 平衡二叉树就是:要么是一棵空树,要么任何一个节点的左右子树高度差的绝对值不超过1. 解法:整个过程为二叉树的后序遍历.对任何一个节点node来说, ...

  3. Dubbo学习系列之十六(ELK海量日志分析框架)

    外卖公司如何匹配骑手和订单?淘宝如何进行商品推荐?或者读者兴趣匹配?还有海量数据存储搜索.实时日志分析.应用程序监控等场景,Elasticsearch或许可以提供一些思路,作为业界最具影响力的海量搜索 ...

  4. SpringBoot整合MybatisPlus3.X之自定义Mapper(十)

    pom.xml <dependencies> <dependency> <groupId>org.springframework.boot</groupId& ...

  5. iOS11 SDK 新特性 CoreML 及swift 小demo

    github代码 如果本博客对您有帮助,希望可以得到您的赞赏! swift 机器学习Core ML的简单调用小demo.完整代码附上: https://github.com/Liuyubao/LYBC ...

  6. 找不到 cucumber.api.cli.Main 的报错解决方案

    最近玩IDEA,发现导入的项目有问题,报了一个“找不到或者不存在cucumber.api.cli.Main”的错误. 后来发现是新版的IDEA在导入时没有提示,以至于我没有配置项目对应的Tomcat服 ...

  7. c#通过libreOffice实现 office文件转pdf文件

    一.安装libreOffice 点击官网下载libreOffice 二.创建一个新的项目LibreOffice 创建一个新的项目,方便后面调用 添加下面代码 public class OfficeCo ...

  8. Flask:第一个Flask项目

    在上一篇文章:Flask:项目的准备工作中,我写了flask项目的准备工作,不清楚创建flask项目需要做哪些准备的朋友可以点击链接看看 1.最简单的Flask项目 代码: from flask im ...

  9. Java 8 - 行为参数化

    前言: 以下内容来源于java8实战一书,我将里面学习的内容做一个整理,方便没空读书的小伙伴快速入门实战java8 正文: 假设一个果农收获了一园的苹果,他需要你根据他的条件筛选出满意的苹果 首先我们 ...

  10. ABAP中时间戳的处理

    UTC(UTC, Universal Time Coordinated,通用协调时)时间戳,分为长时间戳和段时间戳,其中长时间戳餐开始的系统的数据元素TIMESTAMPL,类型为DEC(21,7):而 ...