【CF471E】MUH and Lots and Lots of Segments

题意:给你平面上n条水平或竖直的,端点在整点处的线段。你需要去掉一些线段的一些部分,使得剩下的图形:1.连通,2.无环,3.端点依旧位于整点处。

$n\le 2\times 10^5$

题解:如果把整点看成点的话,那么这题让你求的就是一棵生成树。一棵生成树的边数就是这个连通块内点数-1,所以我们找到最大的连通块将其点数-1就是答案。

具体实现中,我们先进行扫描线,用并查集维护连通性,用线段树快速查找区间中点的数量以及一个点的前驱后继,用set维护所有所在连通块与后继所在连通块不同的点即可。

不知道为什么思考的过程中想到了Kruskal。

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <set>
#define lson x<<1
#define rson x<<1|1
using namespace std;
const int maxn=200010;
typedef long long ll;
int n,m,np,nq;
int f[maxn<<1],org[maxn<<1];
ll ans;
ll ref[maxn<<1];
struct edge
{
int l,r,x;
}p[maxn],q[maxn<<1];
ll s[maxn<<3],sum[maxn<<1];
set<int> st;
set<int>::iterator it;
bool cmp(const edge &a,const edge &b)
{
return (a.x==b.x)?(a.r>b.r):(a.x<b.x);
}
int find(int x)
{
return (f[x]==x)?x:(f[x]=find(f[x]));
}
void updata(int l,int r,int x,int a,int b)
{
s[x]+=b;
if(l==r) return ;
int mid=(l+r)>>1;
if(a<=mid) updata(l,mid,lson,a,b);
else updata(mid+1,r,rson,a,b);
}
int count(int l,int r,int x,int a,int b)
{
if(a>b) return 0;
if(a<=l&&r<=b) return s[x];
int mid=(l+r)>>1;
if(b<=mid) return count(l,mid,lson,a,b);
if(a>mid) return count(mid+1,r,rson,a,b);
return count(l,mid,lson,a,b)+count(mid+1,r,rson,a,b);
}
int find(int l,int r,int x,int a)
{
if(l==r) return l;
int mid=(l+r)>>1;
if(a<=s[lson]) return find(l,mid,lson,a);
return find(mid+1,r,rson,a-s[lson]);
}
inline int pre(int x)
{
int t=count(1,m,1,1,x);
if(t==1) return -1;
return find(1,m,1,t-1);
}
inline int nxt(int x)
{
int t=count(1,m,1,1,x);
if(t==s[1]) return -1;
return find(1,m,1,t+1);
}
inline int rd()
{
int ret=0,f=1; char gc=getchar();
while(gc<'0'||gc>'9') {if(gc=='-') f=-f; gc=getchar();}
while(gc>='0'&&gc<='9') ret=ret*10+(gc^'0'),gc=getchar();
return ret*f;
}
int main()
{
n=rd();
int i,j,a,b,c,d;
ll tmp;
for(i=1;i<=n;i++)
{
a=rd(),b=rd(),c=rd(),d=rd();
if(b==d)
{
q[++nq].x=a,q[nq].l=b,q[nq].r=c-a+1;
q[++nq].x=c,q[nq].l=b,q[nq].r=0;
ref[++m]=b;
}
else
{
p[++np].x=a,p[np].l=b,p[np].r=d;
ref[++m]=b,ref[++m]=d;
ans=max(ans,ll(d-b));
}
}
sort(ref+1,ref+m+1);
for(i=1,j=0,ref[0]=-1<<30;i<=m;i++) if(ref[i]!=ref[j]) ref[++j]=ref[i];
m=j;
for(i=1;i<=nq;i++) q[i].l=lower_bound(ref+1,ref+m+1,q[i].l)-ref,f[i]=i;
for(i=1;i<=np;i++) p[i].l=lower_bound(ref+1,ref+m+1,p[i].l)-ref,p[i].r=lower_bound(ref+1,ref+m+1,p[i].r)-ref;
sort(p+1,p+np+1,cmp);
sort(q+1,q+nq+1,cmp);
for(i=j=1;i<=np;i++)
{
while(j<=nq&&(q[j].x<p[i].x||(q[j].x==p[i].x&&q[j].r)))
{
a=q[j].l,b=q[j].r;
if(b)
{
org[a]=j,sum[j]=b;
st.insert(a);
updata(1,m,1,a,1);
c=pre(a);
if(c!=-1) st.insert(c);
}
else
{
it=st.find(a);
if(it!=st.end()) st.erase(it);
c=pre(a);
if(c!=-1) st.insert(c);
updata(1,m,1,a,-1);
}
j++;
}
tmp=ref[p[i].r]-ref[p[i].l]+1-count(1,m,1,p[i].l,p[i].r);
a=p[i].l-1,c=nxt(a);
if(c<=p[i].r)
{
sum[find(org[c])]+=tmp;
while(1)
{
it=st.lower_bound(p[i].l);
if(it==st.end()||(*it)>p[i].r) break;
a=*it,c=nxt(a);
if(c==-1||c>p[i].r) break;
st.erase(it);
b=find(org[a]),d=find(org[c]);
if(b!=d) sum[d]+=sum[b],f[b]=d;
}
}
}
for(i=1;i<=nq;i++) ans=max(ans,sum[i]-1);
printf("%lld",ans);
return 0;
}

【CF471E】MUH and Lots and Lots of Segments 扫描线+并查集+线段树+set的更多相关文章

  1. Codeforces Round #337 (Div. 2) D. Vika and Segments 线段树 矩阵面积并

    D. Vika and Segments     Vika has an infinite sheet of squared paper. Initially all squares are whit ...

  2. Codeforces Round #337 (Div. 2) D. Vika and Segments 线段树扫描线

    D. Vika and Segments 题目连接: http://www.codeforces.com/contest/610/problem/D Description Vika has an i ...

  3. POJ 1436 Horizontally Visible Segments (线段树&#183;区间染色)

    题意   在坐标系中有n条平行于y轴的线段  当一条线段与还有一条线段之间能够连一条平行与x轴的线不与其他线段相交  就视为它们是可见的  问有多少组三条线段两两相互可见 先把全部线段存下来  并按x ...

  4. Codeforces Round #535 (Div. 3) E2. Array and Segments (Hard version) 【区间更新 线段树】

    传送门:http://codeforces.com/contest/1108/problem/E2 E2. Array and Segments (Hard version) time limit p ...

  5. POJ 1436 Horizontally Visible Segments(线段树)

    POJ 1436 Horizontally Visible Segments 题目链接 线段树处理染色问题,把线段排序.从左往右扫描处理出每一个线段能看到的右边的线段,然后利用bitset维护枚举两个 ...

  6. (中等) POJ 1436 Horizontally Visible Segments , 线段树+区间更新。

    Description There is a number of disjoint vertical line segments in the plane. We say that two segme ...

  7. codeforces 610D D. Vika and Segments(离散化+线段树+扫描线算法)

    题目链接: D. Vika and Segments time limit per test 2 seconds memory limit per test 256 megabytes input s ...

  8. poj 3304 Segments(计算直线与线段之间的关系)

    Segments Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 10921   Accepted: 3422 Descrip ...

  9. POJ 3304 Segments(判断直线与线段是否相交)

    题目传送门:POJ 3304 Segments Description Given n segments in the two dimensional space, write a program, ...

随机推荐

  1. JS_高程5.引用类型(5)Array类型的操作方法

    一.操作方法 1.concat()方法 基于当前数组中的所有项创建一个新数组.具体说,是先创建当前数组的一个副本,然后将接收到的参数添加到这个副本的末尾,最后返回新构建的数组.在没有给concat() ...

  2. Vue(一)创建第一个Vue程序

    一.下载安装nodeJs 基于node.js,利用淘宝npm镜像安装相关依赖.由于国内使用npm会很慢,这里推荐使用淘宝NPM镜像 -- npm install -g cnpm --registry= ...

  3. 移动端适配问题px->rem方法

    移动端web页面适配问题 1.引入插件 github地址:https://github.com/re54k/mobileweb-utilities/blob/master/util/mobile-ut ...

  4. pygame-KidsCanCode系列jumpy-part0-使用sprite

    油管(youtube)上有一个号称"史上最好的pygame教程"(传送门:https://www.youtube.com/watch?v=VO8rTszcW4s&list= ...

  5. CiscoIOUKeygen

    python CiscoIOUKeygen.py | grep -A 1 ‘license’ > iourc

  6. 20180821 Python学习笔记:如何获取当前程序路径

    20180821 Python学习笔记:如何获取当前程序路径 启动的脚本的路径为:D:\WORK\gitbase\ShenzhenHouseInfoCrawler\main.py 当前脚本的路径为:D ...

  7. spring boot使用TestRestTemplate集成测试 RESTful 接口

    这篇文章没什么技术含量,只是单纯的记录一下如何用TestRestTemplate访问受security保护的api,供以后查阅. @Slf4j @RunWith(SpringRunner.class) ...

  8. 几种序列化协议(protobuf,xstream,jackjson,jdk,hessian)相关数据对比

    测试结果 序列化数据对比 bytes字节数对比 具体的数字:   protobuf jackson xstream Serializable hessian2 hessian2压缩 hessian1 ...

  9. 分析轮子(九)- Cloneable.java

    注:玩的是JDK1.7版本 一:Cloneable.java 接口也是标记接口,所以,它没有任何方法和属性,实现此接口表示的意思是:可以调用 Object.java 类的 clone() 方法,进行简 ...

  10. 转-编写CGI小结

    由于Carl要用到我的程序,我们便合作工作.但是他写的程序是Python的,我写的程序是Java的,必须得找一种方式进行通信.尽管有Jython这些东西,但是Carl认为还是CGI最简便.于是,前阵子 ...