P1856 [USACO5.5]矩形周长Picture

$len$            $sum$              $num$             $flag\_l$ $flage\_r$分别表示该区间

覆盖长度--整体覆盖次数--覆盖段数--左右端点是否覆盖

将上下边按高度排序不断扫下来

$num$是用来处理高度的

inline void Update(LL now,LL l,LL r){
if(tree[now].sum){
tree[now].num=1,
tree[now].len=r-l+1,
tree[now].flag_l=tree[now].flag_r=1;
}else if(l==r){
tree[now].len=tree[now].num=tree[now].flag_l=tree[now].flag_r=0;
}else{
LL son0=now<<1,son1=son0|1;
tree[now].len=tree[son0].len+tree[son1].len,
tree[now].num=tree[son0].num+tree[son1].num;
if(tree[son0].flag_r&&tree[son1].flag_l)
--tree[now].num;
tree[now].flag_l=tree[son0].flag_l,
tree[now].flag_r=tree[son1].flag_r;
}
}

现在来理解一下更新

如果该区间至少覆盖了一次,显然只覆盖段数为$1$,注意线段树里长度是以段分的

如果没被覆盖

$(1)$叶子节点,全部清空

$(2)$长度$=$左右子树长度 覆盖段数也差不多,再加一下合并

My complete code:

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<iostream>
#include<string>
using namespace std;
typedef long long LL;
const LL maxn=1e5;
inline LL Read(){
LL x=0,f=1; char c=getchar();
while(c<'0'||c>'9'){
if(c=='-') f=-1; c=getchar();
}
while(c>='0'&&c<='9'){
x=(x<<3)+(x<<1)+c-'0'; c=getchar();
}return x*f;
}
struct node{
LL l,r,h,val;
}a[maxn];
struct code{
LL len,sum,num,flag_l,flag_r;
}tree[maxn];
LL n,mi,mx,ans,num,last;
inline bool cmp(node x,node y){
return x.h<y.h||(x.h==y.h&&x.val>y.val);
}
inline void Update(LL now,LL l,LL r){
if(tree[now].sum){
tree[now].num=1,
tree[now].len=r-l+1,
tree[now].flag_l=tree[now].flag_r=1;
}else if(l==r){
tree[now].len=tree[now].num=tree[now].flag_l=tree[now].flag_r=0;
}else{
LL son0=now<<1,son1=son0|1;
tree[now].len=tree[son0].len+tree[son1].len,
tree[now].num=tree[son0].num+tree[son1].num;
if(tree[son0].flag_r&&tree[son1].flag_l)
--tree[now].num;
tree[now].flag_l=tree[son0].flag_l,
tree[now].flag_r=tree[son1].flag_r;
}
}
LL ci;
void Add(LL now,LL l,LL r,LL lt,LL rt,LL v){
if(lt<=l&&rt>=r){
tree[now].sum+=v,
Update(now,l,r);
return;
}
LL mid=(l+r)>>1,son0=now<<1,son1=son0|1;
if(lt<=mid)
Add(son0,l,mid,lt,rt,v);
if(rt>mid)
Add(son1,mid+1,r,lt,rt,v);
Update(now,l,r);
}
int main(){
n=Read();
for(LL i=1;i<=n;++i){
LL x1=Read(),y1=Read(),x2=Read(),y2=Read();
mx=max(mx,max(x1,x2)),
mi=min(mi,min(x1,x2)),
a[++num]=(node){x1,x2,y1,1},
a[++num]=(node){x1,x2,y2,-1};
}
if(mi<=0){
for(LL i=1;i<=num;++i)
a[i].l+=-mi+1,
a[i].r+=-mi+1;
mx+=-mi;
}
sort(a+1,a+1+num,cmp);
for(LL i=1;i<=num;++i){
Add(1,1,mx,a[i].l,a[i].r-1,a[i].val);
while(a[i].h==a[i+1].h&&a[i].val==a[i+1].val){
++i,
Add(1,1,mx,a[i].l,a[i].r-1,a[i].val);
}
ans+=abs(last-tree[1].len),
last=tree[1].len,
ans+=2*tree[1].num*(a[i+1].h-a[i].h);
}
printf("%lld",ans);
return 0;
}

  

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

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

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

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

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

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

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

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

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

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

    线段树+扫描线 经典的扫描线问题 首先将一个矩形看作由竖着的两条边和横着的两条边构成 那分成两次考虑,一次考虑竖边,一次考虑横边 首先考虑横边 如图两个矩形,现将横边擦去,留下竖边,将平面划分成3个区 ...

  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. 对ps4 cmask fmask的理解

    这俩都是绑在corlor target上8x8的格子 cmask 做fastclear 这个比较好理解,8x8来表示这个格子是否clear fmask msaa用 provided to suppor ...

  2. mybatis配置mapperLocations多个路径

    <property name="mapperLocations"> <array> <value>classpath*:/mybatis-con ...

  3. GIS可视化——麻点图

    一.引言 目前在客户端绘制POI(Point of Interest,兴趣点)的方式主要是div(Marker的形式).svg.canvas.VML(后边三种就是Vector Layer)几种方式,这 ...

  4. 【Salvation】—— 项目策划&市场分析

    写在前面:这个项目是2017年,我们评选校级创新基金项目的参加作品,小组4人,我为负责人,这个项目现在已经基本完成,目前处于后期收尾阶段. 一.项目的目标.内容及创新之处 1.研究目标 体现人类与自然 ...

  5. MySQL触发器 trigger学习

    触发器:一类特殊的事物.可监视某种数据操作,并触发相关操作(insert/update/delete).表中的某些数据改变,希望同一时候能够引起其他相关数据改变的需求. 作用:变化自己主动完毕某些语句 ...

  6. 【Python】删除字符串的空白

    在程序中,额外的空白可能让人迷惑,对于程序员来说,'python'跟'python '看起来几乎一样,但是对于程序来说,可是千差万别 (lstrip)删除开头空白 >>> Langu ...

  7. Nginx配置SSL安全证书避免启动输入Enter PEM pass phrase

    之前两篇文章已经很好的介绍了Nginx配置SSL的一些情况,配置好的Nginx每次启动都要 输两遍PEM pass phrase,很是不爽,尤其是在服务器重启后,Nginx压根就无法自动启动,必须手动 ...

  8. syslog,rsyslog and syslog-ng

    http://en.wikipedia.org/wiki/Syslog Syslog is a standard for computer message logging. It permits se ...

  9. 宜信开源|微服务任务调度平台SIA-TASK入手实践

    引言 最近宜信开源微服务任务调度平台SIA-TASK,SIA-TASK属于分布式的任务调度平台,使用起来简单方便,非常容易入手,部署搭建好SIA-TASK任务调度平台之后,编写TASK后配置JOB进行 ...

  10. HTML中Select的使用详解

    <html><head><SCRIPT LANGUAGE="JavaScript"><!--//oSelect 列表的底部添加了一个新选项 ...