线段树+扫描线

经典的扫描线问题

首先将一个矩形看作由竖着的两条边和横着的两条边构成

那分成两次考虑,一次考虑竖边,一次考虑横边

首先考虑横边

如图两个矩形,现将横边擦去,留下竖边,将平面划分成3个区域

在代码实现时,可以从左到右记录端点

那么现在想象一条平行于横边的扫描线不断从下往上扫

当扫到一个矩形的下边,将这条矩形的下边包含的区间覆盖

当扫到一个矩形的上边,将这条矩形的上边包含的区间去除覆盖

那么对于这些被竖线划分的区域,建立一棵线段树维护这些区间是否被覆盖

如最底下的矩形下边,覆盖区间2,3

那么如何统计答案

答案就是上一次线段树维护的总区间被覆盖的长度-当前线段树维护的总区间覆盖的长度的绝对值

如上图,扫描到第2边,上一次覆盖2,3,那此处增加为1区间

那么线段树中维护这个区间被完全覆盖的次数和被覆盖的长度即可

#include <iostream>
#include <algorithm>
#include <map>
#include <cstdio>
#define inf (int)1e9
using namespace std;
int n,w,last,ans;
struct node
{
int a,b,c,d;
}p[5100];
struct tree
{
int l,r,sum;
int len;
}sh[100000];
struct edge
{
int l,r,num,kind;
}a[11000];
bool cmp(edge a,edge b)
{
return (a.num<b.num || (a.num==b.num && a.kind>b.kind));//注意如果同一高度的话,先处理下边的
}
void pushup(int x)
{
if (sh[x].sum>0)//如果当前被完全覆盖,那么当前区间被覆盖的长度为r-l+1
sh[x].len=sh[x].r-sh[x].l+1;
else
if (sh[x].l==sh[x].r)//叶子结点
sh[x].len=0;
else
sh[x].len=sh[x+x].len+sh[x+x+1].len;//一般情况
}
void build(int x,int ll,int rr)
{
sh[x].l=ll;
sh[x].r=rr;
sh[x].sum=sh[x].len=0;
if (ll==rr)
return;
int mid;
mid=(ll+rr)>>1;
build(x+x,ll,mid);
build(x+x+1,mid+1,rr);
}
void change(int x,int ll,int rr,int v)
{
if (sh[x].l>=ll && sh[x].r<=rr)
{
sh[x].sum+=v;
pushup(x);
return;
}
int mid;
mid=(sh[x].l+sh[x].r)>>1;
if (ll<=mid)
change(x+x,ll,rr,v);
if (rr>mid)
change(x+x+1,ll,rr,v);
pushup(x);
}
int main()
{
scanf("%d",&n);
for (int i=1;i<=n;i++)
scanf("%d%d%d%d",&p[i].a,&p[i].b,&p[i].c,&p[i].d);
for (int i=1;i<=n;i++)//记录前扫描线要扫到的边
{
w++;
a[w].l=p[i].a;a[w].r=p[i].c;
a[w].kind=1;a[w].num=p[i].b;//下边
w++;
a[w].l=p[i].a;a[w].r=p[i].c;
a[w].kind=-1;a[w].num=p[i].d;//上边
}
build(1,-10000,10000);
sort(a+1,a+1+w,cmp);
last=0;//上一次答案
for (int i=1;i<=w;i++)
{
change(1,a[i].l,a[i].r-1,a[i].kind);
ans+=abs(last-sh[1].len);
last=sh[1].len;
}
w=0;//处理竖边
for (int i=1;i<=n;i++)
{
w++;
a[w].l=p[i].b;a[w].r=p[i].d;
a[w].kind=1;a[w].num=p[i].a;
w++;
a[w].l=p[i].b;a[w].r=p[i].d;
a[w].kind=-1;a[w].num=p[i].c;
}
build(1,-10000,10000);
sort(a+1,a+1+w,cmp);
last=0;
for (int i=1;i<=w;i++)
{
change(1,a[i].l,a[i].r-1,a[i].kind);
ans+=abs(last-sh[1].len);
last=sh[1].len;
}
printf("%d\n",ans);
}

Luogu P1856 [USACO5.5]矩形周长Picture的更多相关文章

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

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

  2. P1856 [USACO5.5]矩形周长Picture

    P1856 [USACO5.5]矩形周长Picture $len$            $sum$              $num$             $flag\_l$ $flage\_ ...

  3. P1856 [USACO5.5]矩形周长Picture[扫描线]

    题目背景 墙上贴着许多形状相同的海报.照片.它们的边都是水平和垂直的.每个矩形图片可能部分或全部的覆盖了其他图片.所有矩形合并后的边长称为周长. 题目描述 编写一个程序计算周长. 如图1所示7个矩形. ...

  4. 洛谷P1856 [USACO5.5]矩形周长Picture

    题目背景 墙上贴着许多形状相同的海报.照片.它们的边都是水平和垂直的.每个矩形图片可能部分或全部的覆盖了其他图片.所有矩形合并后的边长称为周长. 题目描述 编写一个程序计算周长. 如图1所示7个矩形. ...

  5. [题解]P1856 [USACO5.5]矩形周长Picture

    Loli 考试的题目之一 题目地址 \(N^2\)做法 #include <cstdio> #include <cstring> #define re register #de ...

  6. [USACO5.5] 矩形周长Picture

    https://www.luogu.org/problemnew/show/P1856 1.每个矩形由两条横向边和两条纵向边组成. 2.对于横向边,按纵坐标排序.设当前讨论的边为 A [s , t] ...

  7. luogu1856 [USACO5.5]矩形周长Picture

    看到一坨矩形就要想到扫描线.(poj atantis) 我们把横边竖边分开计算,因为横边竖边其实没有区别,以下论述全为考虑竖边的. 怎样统计一个竖边对答案的贡献呢?答:把这个竖边加入线段树,当前的总覆 ...

  8. Luogu1856 [USACO5.5]矩形周长Picture (线段树扫描线)

    对于横轴,加上与上一次扫描的差值:对于竖轴,加上高度差与区间内不相交线段\(*2\)的积: 难点在pushdown,注意维护覆盖关系.再就注意负数 #include <iostream> ...

  9. 「USACO5.5」矩形周长Picture

    题目描述 墙上贴着许多形状相同的海报.照片.它们的边都是水平和垂直的.每个矩形图片可能部分或全部的覆盖了其他图片.所有矩形合并后的边长称为周长. 编写一个程序计算周长. 如图1所示7个矩形. 如图2所 ...

随机推荐

  1. JavaCV FFmpeg采集摄像头YUV数据

    前阵子使用利用树莓派搭建了一个视频监控平台(传送门),不过使用的是JavaCV封装好的OpenCVFrameGrabber和FFmpegFrameRecorder. 其实在javacpp项目集中有提供 ...

  2. Java安全之Commons Collections1分析(一)

    Java安全之Commons Collections1分析(一) 0x00 前言 在CC链中,其实具体执行过程还是比较复杂的.建议调试前先将一些前置知识的基础给看一遍. Java安全之Commons ...

  3. .NET Standard 系列

    .NET Standard 是一套正式的 .NET API 规范,有望在所有 .NET 实现中推出. 推出 .NET Standard 的背后动机是要提高 .NET 生态系统中的一致性. ECMA 3 ...

  4. 字节码暴力破解censum(老版本)

    声明 事先声明,本文仅提供破解方法以供个人及读者们学习Java字节码,不提倡破解程序. 本文是个人学习掘金小册张师傅的<JVM字节码从入门到精通>后,作为一个实践的记录,并无恶意. 关于c ...

  5. 线程基本使用--Thread内部方法调用start

    一个问题,下面的代码会如何运行 public class TraditionalThread { public static void main(String[] args) { System.out ...

  6. Docker学习:(一)初识Docker

    Docker(容器虚拟化技术)要点(秒级启动) Docker的WWH公式学习 What[是什么]. Why[为什么要用它]. How[怎么用] 1.Docker简介 (1)问题:为什么会有docker ...

  7. 怎样学好 java ?

    浅谈Java的学习之路--怎样学好JAVA ?Java - 近10年来计算机软件发展过程中的传奇,其在众多开发者心中的地位就如"屠龙刀"."倚天剑". Java ...

  8. centos7搭建docker环境

    Docker简介 Docker是一种虚拟化技术,用来将你的应用程序及其依赖的环境一起打包成一个镜像发布,使得在任何地方都能获得相同的运行环境. Docker 是一个开源项目,诞生于 2013 年初,最 ...

  9. go xpath

    package main import ( "fmt" "github.com/antchfx/htmlquery" "net/http" ...

  10. swoft配置连接池

    bean.php 'db' => [ 'class' => Database::class, 'dsn' => 'mysql:dbname=test;host=127.0.0.1', ...