题目让我们维护一个连通无向图,边有边权,支持加边删边和询问从\(x\)到\(y\)的异或最短路。

考虑到有删边这样的撤销操作,那么用线段树分治来实现,用线段树来维护询问的时间轴。

将每一条边的出现时间段标记到线段树上,表示在这一段询问中这条边存在。

异或最短路的处理方法与最大XOR和路径类似,给线段树上每个节点开一个线性基,找出当前所有的环,插入该节点的线性基,查询时任意找出一条从\(x\)到\(y\)的路径,到线性基中查询,即为答案。

具体实现时用可撤销并查集,采用按秩合并来优化,因为路径压缩会破坏树的结构。

具体实现细节看代码吧。

\(code:\)

#include<bits/stdc++.h>
#define maxn 400010
using namespace std;
template<typename T> inline void read(T &x)
{
x=0;char c=getchar();bool flag=false;
while(!isdigit(c)){if(c=='-')flag=true;c=getchar();}
while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=getchar();}
if(flag)x=-x;
}
int n,m,q,num,tree_cnt,edge_cnt,root,top;
int ls[maxn],rs[maxn],pre[maxn],nxt[maxn],ans[maxn];
map<pair<int,int>,int> mp;
struct Edge
{
int x,y,v;
}e[maxn];
struct node
{
int x,y,deep;
}st[maxn];
struct query
{
int x,y;
}qu[maxn];
void build(int l,int r,int &cur)
{
if(!cur) cur=++tree_cnt;
if(l==r) return;
int mid=(l+r)>>1;
build(l,mid,ls[cur]);
build(mid+1,r,rs[cur]);
}
vector<int> v[maxn];
void insert(int L,int R,int l,int r,int id,int cur)
{
if(L<=l&&R>=r)
{
v[cur].push_back(id);
return;
}
int mid=(l+r)>>1;
if(L<=mid) insert(L,R,l,mid,id,ls[cur]);
if(R>mid) insert(L,R,mid+1,r,id,rs[cur]);
}
int a[maxn][40];
void ins(int x,int cur)
{
for(int i=30;i>=0;--i)
{
if(x&(1<<i))
{
if(!a[cur][i])
{
a[cur][i]=x;
break;
}
else x^=a[cur][i];
}
}
}
int get(int x,int cur)
{
for(int i=30;i>=0;--i)
if((x^a[cur][i])<x)
x^=a[cur][i];
return x;
}
int fa[maxn],de[maxn],dis[maxn];
int find(int x)
{
return fa[x]==x?x:find(fa[x]);
}
int xor_dis(int x)
{
return fa[x]==x?dis[x]:dis[x]^xor_dis(fa[x]);
}
void merge(int x,int y,int v)
{
if(de[x]<de[y]) swap(x,y);
st[++top]=(node){x,y,de[x]};
fa[y]=x,dis[y]=v,de[x]=max(de[x],de[y]+1);
}
void del(int id)
{
int x=st[id].x,y=st[id].y;
fa[y]=y,dis[y]=0,de[x]=st[id].deep;
}
void copy(int x,int y)
{
for(int i=0;i<=30;++i) a[y][i]=a[x][i];
}
void dfs(int l,int r,int cur)
{
int now=top,size=v[cur].size();
for(int i=0;i<size;++i)
{
int id=v[cur][i],x=e[id].x,y=e[id].y,v=e[id].v;
v^=xor_dis(x)^xor_dis(y),x=find(x),y=find(y);
if(x==y) ins(v,cur);
else merge(x,y,v);
}
if(l==r)
{
int x=qu[l].x,y=qu[l].y;
ans[l]=get(xor_dis(x)^xor_dis(y),cur);
}
else
{
int mid=(l+r)>>1;
copy(cur,ls[cur]),dfs(l,mid,ls[cur]);
copy(cur,rs[cur]),dfs(mid+1,r,rs[cur]);
}
while(top>now) del(top--);
}
int main()
{
read(n),read(m),edge_cnt=m;
for(int i=1;i<=n;++i) fa[i]=i;
for(int i=1;i<=m;++i)
{
read(e[i].x),read(e[i].y),read(e[i].v);
mp[make_pair(e[i].x,e[i].y)]=i,pre[i]=1,nxt[i]=-1;
}
read(q);
while(q--)
{
int opt,x,y,v;
read(opt);
if(opt==1)
{
read(x),read(y),read(v);
e[++edge_cnt]=(Edge){x,y,v};
mp[make_pair(x,y)]=edge_cnt;
pre[edge_cnt]=num+1,nxt[edge_cnt]=-1;
}
if(opt==2)
{
read(x),read(y);
nxt[mp[make_pair(x,y)]]=num;
}
if(opt==3)
{
read(x),read(y);
qu[++num]=(query){x,y};
}
}
build(1,num,root);
for(int i=1;i<=edge_cnt;++i)
{
if(nxt[i]<0) nxt[i]=num;
if(pre[i]<=nxt[i]) insert(pre[i],nxt[i],1,num,i,root);
}
dfs(1,num,root);
for(int i=1;i<=num;++i)
printf("%d\n",ans[i]);
return 0;
}

题解 CF938G 【Shortest Path Queries】的更多相关文章

  1. CF938G Shortest Path Queries 和 CF576E Painting Edges

    这两道都用到了线段树分治和按秩合并可撤销并查集. Shortest Path Queries 给出一个连通带权无向图,边有边权,要求支持 q 个操作: x y d 在原图中加入一条 x 到 y 权值为 ...

  2. CF938G Shortest Path Queries

    首先只有询问的话就是个WC的题,线性基+生成树搞一搞就行. 进一步,考虑如果修改操作只有加边怎么做. 好像也没有什么变化,只不过需要在线地往线性基里插入东西而已. 删边呢? 注意到线性基这个玩意是不支 ...

  3. $CF938G\ Shortest\ Path\ Queries$ 线段树分治+线性基

    正解:线段树分治+线性基 解题报告: 传送门$QwQ$ 考虑如果只有操作3,就这题嘛$QwQ$ 欧克然后现在考虑加上了操作一操作二 于是就线段树分治鸭 首先线段树叶子节点是询问嘛这个不用说$QwQ$. ...

  4. 【CF938G】Shortest Path Queries(线段树分治,并查集,线性基)

    [CF938G]Shortest Path Queries(线段树分治,并查集,线性基) 题面 CF 洛谷 题解 吼题啊. 对于每个边,我们用一个\(map\)维护它出现的时间, 发现询问单点,边的出 ...

  5. Codeforces 938G Shortest Path Queries [分治,线性基,并查集]

    洛谷 Codeforces 分治的题目,或者说分治的思想,是非常灵活多变的. 所以对我这种智商低的选手特别不友好 脑子不好使怎么办?多做题吧-- 前置知识 线性基是你必须会的,不然这题不可做. 推荐再 ...

  6. CF 938G Shortest Path Queries

    又到了喜闻乐见的写博客清醒时间了233,今天做的依然是线段树分治 这题算是经典应用了吧,假的动态图(可离线)问题 首先不难想到对于询问的时间进行线段树分治,这样就可以把每一条边出现的时间区间扔进线段树 ...

  7. Shortest Path(思维,dfs)

    Shortest Path  Accepts: 40  Submissions: 610  Time Limit: 4000/2000 MS (Java/Others)  Memory Limit: ...

  8. HDU 5636 Shortest Path 暴力

    Shortest Path 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5636 Description There is a path graph ...

  9. Shortest Path

    Shortest Path Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)T ...

随机推荐

  1. elasticsearchBouncing Results问题

    bouncing results问题,两个document排序,field值相同:不同的shard上,可能排序不同:每次请求轮询打到不同的replica shard上:每次页面上看到的搜索结果的排序都 ...

  2. yqq命令

    用 apt-get install 安装时 ,会有一个提问,问是否继续,需要输入 yes.使用 yqq 的话,就没有这个提问了,自动 yes . 例如下面的更新和安装nginx: apt-get -y ...

  3. QtableWidget用法流程

    QtableWidget用法流程 ​ 作者:流火 日期:2020/5/10 QTableWidget的基本构造函数 QTableWidget 是QTableview的子类.主要去呗是QTableVie ...

  4. Java面向对象(一)类和对象

    面向过程和面向对象的区别 面向过程,强调的是功能行为,是将实现一个功能的步骤编写在一个函数中,以函数为最小单位. 面向对象,是将需要的功能封装进一个对象中,使一个对象具有很多的功能特征,以类/对象为最 ...

  5. Windows 10 搭键开源IDS-Snort,简单过滤注入

    关于Snort snort有三种工作模式:嗅探器.数据包记录器.网络入侵检测系统.嗅探器模式仅仅是从网络上读取数据包并作为连续不断的流显示在终端上.数据包记录器模式把数据包记录到硬盘上.网路入侵检测模 ...

  6. 这些 CSS 命名规范将省下你大把调试时间

    我听说很多开发者厌恶 CSS.而在我的经验中,这往往是由于他们并没有花时间来学习 CSS. CSS 算不上是最优美的『语言』,但迄今二十多年来,它都是美化 web 举足轻重的工具.从这点来说,也还算不 ...

  7. HTML5全局属性汇总

    局部属性和全局属性 局部属性:有些元素能规定自己的属性,这种属性称为局部属性.比如link元素,它具有的局部属性有href. rel. hreflang. media. type. sizes这六个. ...

  8. Java嵌套类,内部类和外部类

    1.嵌套类,内部类 嵌套类是指被定义在一个类内部的类: JAVA的嵌套类有很多种类:1.静态成员类:2.非静态成员类:3.匿名类:4.局部类:其中,除了静态成员类之外,其他的都是内部类,因为静态成员类 ...

  9. NOI Online #3 提高组 T1水壶 题解

    题目描述 有 n 个容量无穷大的水壶,它们从 1∼n 编号,初始时 i 号水壶中装有 Ai 单位的水. 你可以进行不超过 k 次操作,每次操作需要选择一个满足 1≤x≤n−1 的编号 x,然后把 x ...

  10. pandas | 使用pandas进行数据处理——DataFrame篇

    本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是pandas数据处理专题的第二篇文章,我们一起来聊聊pandas当中最重要的数据结构--DataFrame. 上一篇文章当中我们介绍了 ...