之前学了一下线段树分治,这还是第一次写。思想其实挺好理解,即离线后把一个操作影响到的时间段拆成线段树上的区间,并标记永久化。之后一块处理,对于某个节点表示的时间段,影响到他的就是该节点一直到线段树根的所有操作。(语死早)这样可以把操作的插入和删除改为只有插入。

具体到这题,由于并查集没法删除边,我们考虑线段树分治。之后要考虑的问题就是如何用并查集判断是否为二分图,也即是否含奇环。假设现在图中有一个偶环,若给偶环两点加了一条边,可以发现无论去掉原偶环上哪一条边都不会改变新出现环的奇偶性。于是我们只要用并查集维护出每个点到根的距离(按秩合并),从而维护出任意两点距离的奇偶性,若加入一条环边则判断是否会构成奇环,有奇环直接退出,没有则扔掉这条边不管。每次处理完线段树一个节点后要把操作还原,以避免影响其他无关节点。这个可以在操作时用栈记录,退出时还原。

这个题就做完了。然而代码能力极差的我发现自己连并查集都差点不会写了。以及,图中会有自环,有自环就不是二分图,我开始竟然反方向判掉了……还有会有出现时刻与消失时刻相同的边,RE了好几发。总之调到吐血,早日退役保平安。

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<vector>
#include<stack>
using namespace std;
int read()
{
int x=,f=;char c=getchar();
while (c<''||c>'') {if (c=='-') f=-;c=getchar();}
while (c>=''&&c<='') x=(x<<)+(x<<)+(c^),c=getchar();
return x*f;
}
#define N 100010
int n,m,T,L[N<<],R[N<<],fa[N],deep[N],dis[N],v[N];
struct edge{int x,y;};
struct data{int x,fa,deep,v;};
vector<edge> tree[N<<];
stack<data> undo[N<<];
bool ans[N];
void build(int k,int l,int r)
{
L[k]=l,R[k]=r;
if (l==r) return;
int mid=l+r>>;
build(k<<,l,mid);
build(k<<|,mid+,r);
}
void add(int k,int l,int r,edge e)
{
if (L[k]==l&&R[k]==r) {tree[k].push_back(e);return;}
int mid=L[k]+R[k]>>;
if (r<=mid) add(k<<,l,r,e);
else if (l>mid) add(k<<|,l,r,e);
else add(k<<,l,mid,e),add(k<<|,mid+,r,e);
}
int find(int x)
{
if (fa[x]==x) return x;
int p=find(fa[x]);
dis[x]=dis[fa[x]]^v[x];
return p;
}
void merge(int k,int x,int y,int p)
{
if (deep[x]<deep[y]) swap(x,y);
data a;
a.x=y;a.fa=x;a.deep=deep[x];a.v=v[y];
undo[k].push(a);
fa[y]=x;v[y]=p;dis[y]=dis[x]^p;
if (deep[y]==deep[x]) deep[x]++;
}
void solve(int k)
{
int s=tree[k].size();
bool flag=;
for (int i=;i<s;i++)
{
if (tree[k][i].x==tree[k][i].y) {flag=;break;}
int p=find(tree[k][i].x),q=find(tree[k][i].y);
if (p!=q) merge(k,p,q,dis[tree[k][i].x]^dis[tree[k][i].y]^);
else if (dis[tree[k][i].x]^dis[tree[k][i].y]^) {flag=;break;}
}
if (L[k]<R[k]&&!flag) solve(k<<),solve(k<<|);
else if (flag) for (int i=L[k];i<=R[k];i++) ans[i]=;
while (!undo[k].empty())
{
data a=undo[k].top();
fa[a.x]=a.x;
deep[a.fa]=a.deep;
dis[a.x]=v[a.x]=a.v;
undo[k].pop();
}
}
int main()
{
freopen("bzoj4025.in","r",stdin);
freopen("bzoj4025.out","w",stdout);
n=read(),m=read(),T=read();
build(,,T);
for (int i=;i<=m;i++)
{
edge e;e.x=read(),e.y=read();
int s=read()+,t=read();
if (s<=t) add(,s,t,e);
}
for (int i=;i<=n;i++) fa[i]=i,deep[i]=v[i]=dis[i]=;
solve();
for (int i=;i<=T;i++)
if (ans[i]) printf("No\n");
else printf("Yes\n");
fclose(stdin);fclose(stdout);
return ;
}

BZOJ4025 二分图(线段树分治+并查集)的更多相关文章

  1. bzoj4025二分图(线段树分治 并查集)

    /* 思维难度几乎没有, 就是线段树分治check二分图 判断是否为二分图可以通过维护lct看看是否链接出奇环 然后发现不用lct, 并查集维护奇偶性即可 但是复杂度明明一样哈 */ #include ...

  2. [BZOJ4025]二分图(线段树分治,并查集)

    4025: 二分图 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 2191  Solved: 800[Submit][Status][Discuss] ...

  3. 2018.09.30 bzoj4025: 二分图(线段树分治+并查集)

    传送门 线段树分治好题. 这道题实际上有很多不同的做法: cdq分治. lct. - 而我学习了dzyo的线段树分治+并查集写法. 所谓线段树分治就是先把操作分成lognlognlogn个连续不相交的 ...

  4. BZOJ3237:[AHOI2013]连通图(线段树分治,并查集)

    Description Input Output Sample Input 4 5 1 2 2 3 3 4 4 1 2 4 3 1 5 2 2 3 2 1 2 Sample Output Connec ...

  5. Bzoj1018/洛谷P4246 [SHOI2008]堵塞的交通(线段树分治+并查集)

    题面 Bzoj 洛谷 题解 考虑用并查集维护图的连通性,接着用线段树分治对每个修改进行分治. 具体来说,就是用一个时间轴表示图的状态,用线段树维护,对于一条边,我们判断如果他的存在时间正好在这个区间内 ...

  6. BZOJ 1018: [SHOI2008]堵塞的交通traffic(线段树分治+并查集)

    传送门 解题思路 可以离线,然后确定每个边的出现时间,算这个排序即可.然后就可以线段树分治了,连通性用并查集维护,因为要撤销,所以要按秩合并,时间复杂度\(O(nlog^2 n)\) 代码 #incl ...

  7. BZOJ3237 AHOI2013连通图(线段树分治+并查集)

    把查询看做是在一条时间轴上.那么每条边都有几段存在时间.于是线段树分治就好了. 然而在bzoj上t掉了,不知道是常数大了还是写挂了. 以及brk不知道是啥做数组名过不了编译. #include< ...

  8. BZOJ4025 二分图 线段树分治、带权并查集

    传送门 如果边不会消失,那么显然可以带权并查集做(然后发现自己不会写带权并查集) 但是每条边有消失时间.这样每一条边产生贡献的时间对应一段区间,故对时间轴建立线段树,将每一条边扔到线段树对应的点上. ...

  9. bzoj4025-二分图【线段树分治,并查集】

    正题 题目链接:https://darkbzoj.tk/problem/4025 题目大意 \(n\)个点\(m\)条边,每条边会在一个\(T\)以内的时间段内出现,对于任意一个\(T\)以内的时刻求 ...

随机推荐

  1. github打开慢,甚至打不开

    有人使用github后,在某些网络下发现打开慢,甚至打不开,这都是因为他是国外站:目前互联网的连接机制导致超过一定的路由节点的连接就会出现这个问题,解决办法就是直接告诉本机ip.不要先层层询问域名转i ...

  2. LOJ6089 小Y的背包计数问题 背包、根号分治

    题目传送门 题意:给出$N$表示背包容量,且会给出$N$种物品,第$i$个物品大小为$i$,数量也为$i$,求装满这个背包的方案数,对$23333333$取模.$N \leq 10^5$ $23333 ...

  3. LinqPad的变量比较功能

    LinqPad是一个非常方便的C#工具(有免费版和收费版). 今天发现它的变量比较功能真是方便啊.且看3行代码产生如下结果: 说明:图中两个变量的成员属性值分别用红色和绿色背景标注:图很长,只截取了一 ...

  4. java json字符串传递给 js 时 特殊字符转义错误 研究

    一些换行 回车等符号需要转义 主要注意 单引号 与双引号. 一 如果js以 双引号接收字符串 则转单引号 "  至 \" 否则js报错 二 如果js以 单引号接收字符串 则转单引号 ...

  5. Session帮助类

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.W ...

  6. HDU 3400

    一道很适合练习三分的题目三分套三分强不强 题意:给你平面上两条平行线段\(AB\)和\(CD\),一个人要从\(A\)走到\(D\),他在线段\(AB\)上的速度为\(P\),在\(CD\)上的速度为 ...

  7. TDD、BDD、ATDD、DDD 软件开发模式

    TDD.BDD.ATDD.DDD 软件开发模式 四个开发模式意思: TDD:测试驱动开发(Test-Driven Development) BDD:行为驱动开发(Behavior Driven Dev ...

  8. Ionic 1 & 2 开发常见问题 Q&A

    原文发表于我的技术博客 本文分享了在 Ionic 1 & 2 版本开发过程中常见问题的一些 Q&A,供慕课网同学或其他朋友参考. 原文发表于我的技术博客 1. 版本的问题 Ionic ...

  9. combox的基本应用

    easyui-combox:控件的初始化: 可以在其中进行文字的筛选功能(过滤), 动态加载数据的方法. <!DOCTYPE html><html lang="en&quo ...

  10. (转)SqlDateTime 溢出。必须介于 1/1/1753 12:00:00 AM 和 12/31/9999 11:59:59 PM之间

    原因: 出现这种问题多半是因为你插入或者更新数据库时,datetime字段值为空默认插入0001年01月01日造成datetime类型溢出. 传给数据库表的时间类型值是null值.这里的null指的是 ...