首先只有询问的话就是个WC的题,线性基+生成树搞一搞就行。

进一步,考虑如果修改操作只有加边怎么做。

好像也没有什么变化,只不过需要在线地往线性基里插入东西而已。

删边呢?

注意到线性基这个玩意是不支持删除操作的。

对于这种不好删除的的东西有种不错的解决方法,就是线段树分治。

把每个操作劈成logn个区间以后来搞一下。

按照线段树分治的套路,通过遍历整棵线段树来获得答案。

发现需要一个可以动态维护支持加/删边的生成树的东西。

能做到这个的无非就两个数据结构,按秩合并的并查集和LCT。

这里使用按秩合并的并查集。

注意:

这里之所以利用线段树分治的原因是,使用线段树结构可以在每个节点都维护出对应的线性基。

儿子的线性基可以由线段树上的父亲的线性基加上自身节点内的操作来得到,由于线性基很小,可以快速下传,因此可以这么搞。

剩下的就是码码码了

#include<iostream>
#include<cctype>
#include<cstdio>
#include<cstring>
#include<string>
#include<cmath>
#include<ctime>
#include<map>
#include<vector>
#include<cstdlib>
#include<algorithm>
#define N 440000
#define L 1000000
#define M 6600000
#define eps 1e-7
#define inf 1e9+7
#define db double
#define ll long long
#define ldb long double
using namespace std;
inline int read()
{
char ch=0;
int x=0,flag=1;
while(!isdigit(ch)){ch=getchar();if(ch=='-')flag=-1;}
while(isdigit(ch)){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();}
return x*flag;
}
struct link{int x,y,z;};
bool operator<(link a,link b){if(a.x!=b.x)return a.x<b.x;return a.y<b.y;}
map<link,int>S;
map<link,int>::iterator it;
int n,m,qnum,now,cnt,tot,ans[N];
struct ques{int x,y,id;}q[N];
struct node{link s;int l,r;}p[N*2];
//find-union-set
int top,f[N],v[N],sz[N],sk[M];
int find(int x){if(x!=f[x])return find(f[x]);else return f[x];}
int value(int x){if(x!=f[x])return value(f[x])^v[x];else return 0;}
bool merge(link s)
{
int a=s.x,b=s.y,x=find(a),y=find(b),z=s.z;
if(x==y)return false;
if(sz[x]>sz[y])swap(x,y),swap(a,b);
v[x]=value(a)^value(b)^z;
f[x]=y;sz[y]+=sz[x];sk[++top]=x;
return true;
}
void erase()
{
int x=sk[top--];
sz[f[x]]-=sz[x];f[x]=x;v[x]=0;
}
//linear basis
struct pot
{
int s[33];
void insert(int x)
{
for(int i=31;i>=0;i--)if(1<<i&x)
{
if(!s[i]){s[i]=x;break;}
else x^=s[i];
}
}
int query(int x)
{
for(int i=31;i>=0;i--)if(1<<i&x)x^=s[i];
return x;
}
}sp[200];
//Segment_Tree
#define lson o<<1
#define rson o<<1|1
#define mid ((l+r)>>1)
vector<link>V[N*4];
void optset(int o,int l,int r,int ql,int qr,link s)
{
if(ql<=l&&r<=qr){V[o].push_back(s);return;}
if(ql<=mid)optset(lson,l,mid,ql,qr,s);
if(qr>mid)optset(rson,mid+1,r,ql,qr,s);
}
void solve(int o,int l,int r,int t)
{
if(l>r)return;
int times=0,len=V[o].size();
for(int i=0;i<len;i++)
if(merge(V[o][i]))times++;
else sp[t].insert(value(V[o][i].x)^value(V[o][i].y)^V[o][i].z);
if(l==r)
{
while(now<tot&&q[now+1].id==l)
{
now++;
ans[q[now].id]=sp[t].query(value(q[now].x)^value(q[now].y));
}
}
else
{
for(int i=31;i>=0;i--)sp[t+1].s[i]=sp[t].s[i];solve(lson,l,mid,t+1);
for(int i=31;i>=0;i--)sp[t+1].s[i]=sp[t].s[i];solve(rson,mid+1,r,t+1);
}
for(int i=0;i<times;i++)erase();
}
int main()
{
n=read();m=read();
for(int i=1;i<=m;i++)
{
link o;
o.x=read();o.y=read();o.z=read();
if(o.x>o.y)swap(o.x,o.y);
S.insert(pair<link,int>(o,0));
}
qnum=read();
for(int i=1;i<=qnum;i++)
{
int flag=read();
if(flag==1)
{
link o;
o.x=read();o.y=read();o.z=read();
if(o.x>o.y)swap(o.x,o.y);
S.insert(pair<link,int>(o,i));
}
if(flag==2)
{
link o;
o.x=read();o.y=read();
it=S.find(o);
p[++cnt]=(node){it->first,it->second,i};
S.erase(it);
}
if(flag==3)
{
q[++tot].id=i;q[tot].x=read();q[tot].y=read();
if(q[tot].x>q[tot].y)swap(q[tot].x,q[tot].y);
}
}
for(it=S.begin();it!=S.end();it++)p[++cnt]=(node){it->first,it->second,m};
for(int i=1;i<=cnt;i++)optset(1,0,qnum,p[i].l,p[i].r,p[i].s);
for(int i=1;i<=n;i++)f[i]=i,v[i]=0,sz[i]=1;
solve(1,0,m,1);
for(int i=1;i<=tot;i++)printf("%d\n",ans[q[i].id]);
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$ 线段树分治+线性基

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

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

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

  4. 题解 CF938G 【Shortest Path Queries】

    题目让我们维护一个连通无向图,边有边权,支持加边删边和询问从\(x\)到\(y\)的异或最短路. 考虑到有删边这样的撤销操作,那么用线段树分治来实现,用线段树来维护询问的时间轴. 将每一条边的出现时间 ...

  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. Shortest Path

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

  9. 74(2B)Shortest Path (hdu 5636) (Floyd)

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

随机推荐

  1. Jackson基础

    一.所需jar包: jackson-core-x.x.x-rc4.jar.jackson-databind-x.x.x-rc4.jar.jackson-annotations-x.x.x-rc4.ja ...

  2. Js基础知识4-函数的三种创建、四种调用(及关于new function()的解释)

    在js中,函数本身属于对象的一种,因此可以定义.赋值,作为对象的属性或者成为其他函数的参数.函数名只是函数这个对象类的引用. 函数定义 // 函数的三种创建方法(定义方式) function one( ...

  3. ES6学习--对象属性的遍历

    ES6一共有5种方法可以遍历对象的属性. (1)for...in for...in循环遍历对象自身的和继承的可枚举属性(不含Symbol属性). (2)Object.keys(obj) Object. ...

  4. Linux学习笔记之passwd:Authentication token manipulation error_错误的解决办法

    如果在linux中,不管是root用户还是普通用户登录后,修改自己的密码,出现—passwd:Authentication token manipulation error—错误的解决办法: root ...

  5. 08: CORS实现跨域请求

    目录: 1.1 cors跨域请求介绍 1.2 使用tornado实现 复杂请求 1.3 Django中使用django-cors-headers解决跨域问题 1.1 cors跨域请求介绍返回顶部 1. ...

  6. 20145307陈俊达_安卓逆向分析_APKtools分析smail

    20145307陈俊达_安卓逆向分析_APKtools分析smail 引言 真刺激呢!到了第二篇博客了,难度开始加大,之前是简单的dex2jar和有图形界面的jd-gui,现在来隆重介绍强大的反汇编工 ...

  7. linux内核分析 第一周

    计算机是如何工作的 冯·诺依曼理论的要点是: 数字计算机的数制采用二进制:计算机应该按照程序顺序执行. 冯·诺依曼体系结构 根据冯·诺依曼体系结构构成的计算机,必须具有如下功能:把需要的程序和数据送至 ...

  8. HttpClient 的使用

    HttpClient使用: maven: <!-- https://mvnrepository.com/artifact/org.apache.httpcomponents/httpclient ...

  9. LightOJ 1393 Crazy Calendar(博弈)题解

    题意:r*c方格中,每个格子有一定石子,每次移动每格任意数量石子,只能向下或者向右动一格,不能移动为败 思路:显然是Nim,到右下曼哈顿距离为偶数的不用管,因为先手动一下后手动一下最后移到右下后还是先 ...

  10. 源码编译安装keepalived

    首先到官网下载需要的包:http://www.keepalived.org/download.html [root@localhost local]# .tar.gz [root@localhost ...