题目

给出一个连通带权无向图,边有边权,要求支持 \(q\) 个操作:

  1. \(x\) \(y\) \(d\) 在原图中加入一条 \(x\) 到 \(y\) 权值为 \(b\) 的边

  2. \(x\) \(y\) 把图中 \(x\) 到 \(y\) 的边删掉

  3. \(x\) \(y\) 表示询问 \(x\) 到 \(y\) 的异或最短路

保证任意操作后原图连通无重边自环且操作均合法

\(n,m,q≤200000\)


分析

删除很难做,考虑线段树分治,

那么题目就转换成求某一时间点,

无向图的异或最短路,先构建一棵生成树,

那么环上边等于是将生成树异或值抵消掉,

用线性基求异或最小值

那么在并查集的基础上维护生成树点到根的距离即可


代码

#include <cstdio>
#include <cctype>
#include <vector>
#include <cstring>
#include <map>
#define rr register
using namespace std;
const int N=200011; vector<int>K[N<<2]; struct five{int x,y,w,l,r;}e[N<<1];
struct rec{int x,y;}q[N]; map<pair<int,int>,int>uk;
int n,m,T,dep[N],f[N],d[N],stac[N],stad[N],tac[N],tad[N],staD[N],taD[N],tot,tod,toD,ans[N];
inline signed iut(){
rr int ans=0; rr char c=getchar();
while (!isdigit(c)) c=getchar();
while (isdigit(c)) ans=(ans<<3)+(ans<<1)+(c^48),c=getchar();
return ans;
}
inline void print(int ans){
if (ans>9) print(ans/10);
putchar(ans%10+48);
}
struct Vector_Space{
int re[30];
inline void BUILD(){memset(re,0,sizeof(re));}
inline void Insert(int x){
for (rr int i=29;~i;--i)
if ((x>>i)&1){
if (re[i]) x^=re[i];
else {re[i]=x; return;}
}
}
inline signed query(int x){
for (rr int i=29;~i;--i)
if ((x^re[i])<x) x^=re[i];
return x;
}
};
inline void update(int k,int l,int r,int x,int y,int z){
if (l==x&&r==y) {K[k].push_back(z); return;}
rr int mid=(l+r)>>1;
if (y<=mid) update(k<<1,l,mid,x,y,z);
else if (x>mid) update(k<<1|1,mid+1,r,x,y,z);
else update(k<<1,l,mid,x,mid,z),update(k<<1|1,mid+1,r,mid+1,y,z);
}
inline signed getf(int u){return f[u]==u?u:getf(f[u]);}
inline signed getd(int u){return f[u]==u?0:(d[u]^getd(f[u]));}
inline void dfs(int k,int l,int r,Vector_Space H){
rr int len=K[k].size(),Tot=tot,Tod=tod,ToD=toD;
for (rr int i=0;i<len;++i){
rr five t=e[K[k][i]];
rr int fa=getf(t.x),fb=getf(t.y);
rr int W=getd(t.x)^getd(t.y)^t.w;
if (fa==fb){H.Insert(W); continue;}
if (dep[fa]>dep[fb]) fa^=fb,fb^=fa,fa^=fb;
staD[++toD]=fa,taD[toD]=d[fa],d[fa]=W;
if (dep[fa]==dep[fb]){
stad[++tod]=fb,tad[tod]=dep[fb];
++dep[fb];
}
stac[++tot]=fa,tac[tot]=f[fa],f[fa]=fb;
}
rr int mid=(l+r)>>1;
if (l==r) ans[l]=H.query(getd(q[l].x)^getd(q[l].y));
else dfs(k<<1,l,mid,H),dfs(k<<1|1,mid+1,r,H);
for (;tot>Tot;--tot) f[stac[tot]]=tac[tot];
for (;tod>Tod;--tod) dep[stad[tod]]=tad[tod];
for (;toD>ToD;--toD) d[staD[toD]]=taD[toD];
}
signed main(){
n=iut(),m=iut();
for (rr int i=1;i<=n;++i) f[i]=i,dep[i]=1;
for (rr int i=1;i<=m;++i){
rr int x=iut(),y=iut(),w=iut();
if (x>y) x^=y,y^=x,x^=y;
e[i]=(five){x,y,w,0,-2},uk[make_pair(x,y)]=i;
}
for (rr int Q=iut();Q;--Q){
rr int opt=iut(),x=iut(),y=iut();
if (x>y) x^=y,y^=x,x^=y;
if (opt==1)
e[++m]=(five){x,y,iut(),T,-2},
uk[make_pair(x,y)]=m;
else if (opt==2){
rr int now=uk[make_pair(x,y)];
e[now].r=T-1;
}else q[T++]=(rec){x,y};
}
for (rr int i=1;i<=m;++i) if (e[i].r==-2) e[i].r=T;
for (rr int i=1;i<=m;++i)
if (e[i].l<=e[i].r) update(1,0,T,e[i].l,e[i].r,i);
rr Vector_Space H; H.BUILD(),dfs(1,0,T,H);
for (rr int i=0;i<T;++i) print(ans[i]),putchar(10);
return 0;
}

#线段树分治,线性基,并查集#CF938G Shortest Path Queries的更多相关文章

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

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

  2. BZOJ.4184.shallot(线段树分治 线性基)

    BZOJ 裸的线段树分治+线性基,就是跑的巨慢_(:з」∠)_ . 不知道他们都写的什么=-= //41652kb 11920ms #include <map> #include < ...

  3. Codeforces 938G 线段树分治 线性基 可撤销并查集

    Codeforces 938G Shortest Path Queries 一张连通图,三种操作 1.给x和y之间加上边权为d的边,保证不会产生重边 2.删除x和y之间的边,保证此边之前存在 3.询问 ...

  4. LOJ 2312(洛谷 3733) 「HAOI2017」八纵八横——线段树分治+线性基+bitset

    题目:https://loj.ac/problem/2312 https://www.luogu.org/problemnew/show/P3733 原本以为要线段树分治+LCT,查了查发现环上的值直 ...

  5. 【luogu3733】【HAOI2017】 八纵八横 (线段树分治+线性基)

    Descroption 原题链接 给你一个\(n\)个点的图,有重边有自环保证连通,最开始有\(m\)条固定的边,要求你支持加边删边改边(均不涉及最初的\(m\)条边),每一次操作都求出图中经过\(1 ...

  6. 【线段树分治 线性基】luoguP3733 [HAOI2017]八纵八横

    不知道为什么bzoj没有HAOI2017 题目描述 Anihc国有n个城市,这n个城市从1~n编号,1号城市为首都.城市间初始时有m条高速公路,每条高速公路都有一个非负整数的经济影响因子,每条高速公路 ...

  7. BZOJ4184:shallot(线段树分治,线性基)

    Description 小苗去市场上买了一捆小葱苗,她突然一时兴起,于是她在每颗小葱苗上写上一个数字,然后把小葱叫过来玩游戏. 每个时刻她会给小葱一颗小葱苗或者是从小葱手里拿走一颗小葱苗,并且 让小葱 ...

  8. bzoj 4184 shallot——线段树分治+线性基

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4184 本来想了可持久化trie,不过空间是 nlogn (出一个节点的时候把 tot 复原就 ...

  9. bzoj 4184: shallot (线段树维护线性基)

    题面 \(solution:\) 这一题绝对算的上是一道经典的例题,它向我们诠释了一种新的线段树维护方式(神犇可以跳过了).像这一类需要加入又需要维护删除的问题,我们曾经是遇到过的像莫对,线段树... ...

  10. Wanafly 挑战赛 14 E 无效位置 (线性基+并查集)

    Wanafly 挑战赛 14 E 无效位置 (线性基+并查集) 传送门:https://ac.nowcoder.com/acm/contest/81/#question 题意: n个数,m次操作 一个 ...

随机推荐

  1. [BUUCTF][Web][极客大挑战 2019]LoveSQL 1

    打开靶机url,页面显示有两个输入框,框中输入123',发现两个框都有sql注入问题 爆出一下错误 You have an error in your SQL syntax; check the ma ...

  2. Windows系统下的输入法选择

    总共用过5款输入法:搜狗拼音输入法,QQ拼音输入法,谷歌拼音输入法,手心输入法,小狼毫. 搜狗输入法功能最强大,词库也很全,基本上对于盲打的输入纠错很准确,但是因为后台会启动多个服务,会占很多内存资源 ...

  3. drf中认证源码流程

    drf中认证流程 首先通过导入from rest_framework.views import APIView,然后通过ctrl+鼠标右键进入到APIView类中,apiview中定义了许多方法,我们 ...

  4. 【LeetCode二叉树#15】二叉搜索树的最小绝对差(巩固迭代中序遍历#2)

    二叉搜索树的最小绝对差(迭代法中序遍历巩固) 力扣题目链接(opens new window) 给你一棵所有节点为非负值的二叉搜索树,请你计算树中任意两节点的差的绝对值的最小值. 示例: 提示:树中至 ...

  5. 【Azure ACR+App Service】ACR WebHook请求App Service时遇见 401 Unauthorized

    问题描述 App Service 支持从ACR中直接拉取镜像,并且可以配置持续部署(Continuous Deployment), 它是通过在ACR中添加一个Webhook,然后发送POST请求到 & ...

  6. 【Azure 应用服务】使用Docker Compose创建App Service遇见"Linux Version is too long. It cannot be more than 4000 characters"错误

    问题描述 使用Docker Compose方式合并多个镜像(Images)文件,然后部署到App Service中,结果失败.报错 Linux Version 太长,不能超过4000个字符. 错误消息 ...

  7. mysql-批量修改表的主键id,修改成联合主键

    1.sql脚本 一. 通过sql脚本,查出所有表的功能,并编写插入修改的联合主键,sql select concat('ALTER table ', TABLE_NAME, ' DROP PRIMAR ...

  8. 上来就对标 20k Star 的开源项目,是自不量力还是后起之秀?

    先来一段紧箍咒:nvm.fvm.gvm.sdkman.fnm.n.g.rvm.jenv.phpbrew.rustup.swiftenv.pyenv.rbenv... 这些都是用来解决编程语言多版本管理 ...

  9. Codeforces Round #851 (Div. 2) 题解

    Codeforces Round #851 (Div. 2) 题解 A. One and Two 取 \(\log_2\),变成加号,前缀和枚举 \(s[i]=\dfrac{s[n]}{2}\). B ...

  10. vim 学习总结

    vim 学习总结 一.介绍 vi 编辑器是所有 Unix 及 Linux 系统下标准的编辑器,类似于 Windows 系统下的 notepad(记事本)编辑器.在 Unix 及 Linux 系统的任何 ...