Code:

#include <map>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <string>
using namespace std; void setIO(string a){
freopen((a+".in").c_str(),"r",stdin);
freopen((a+".out").c_str(),"w",stdout);
} #define N 30005
#define Q 40005
#define M 100005 int n,m,cnt=1;
int head[N];
int to[M<<1];
int nex[M<<1]; int siz[N];
int f[N];
int son[N];
int dep[N];
int uu[M<<1];
int vv[M<<1];
bool tag[M<<1];
bool vis[N]; void addedge(int u,int v){
nex[++cnt]=head[u],head[u]=cnt,to[cnt]=v;
uu[cnt]=u,vv[cnt]=v;
} void dfs1(int u,int fa,int depth){
siz[u]=1,f[u]=fa,vis[u]=1,dep[u]=depth+1;
for(int v=head[u];v;v=nex[v]){
if(!vis[to[v]]&&to[v]!=fa) {
tag[v]=tag[v^1]=1;
dfs1(to[v],u,depth+1);
siz[u]+=siz[to[v]];
if(siz[to[v]]>siz[son[u]]) son[u]=to[v];
}
}
} int A[N];
int top[N];
int cnt2=0; void dfs2(int u,int tp){
top[u]=tp;
A[u]=++cnt2;
if(son[u])dfs2(son[u],tp);
for(int v=head[u];v;v=nex[v])
if(to[v]!=son[u]&&to[v]!=f[u]&&tag[v])dfs2(to[v],to[v]);
//if(!son[u]) son[u]=u;
} int sumv[N<<2];
int lazy[N<<2]; #define lson (o<<1)
#define rson (o<<1)|1 void pushup(int o){
sumv[o]=sumv[lson]+sumv[rson];
if(lazy[lson]&&lazy[rson]) lazy[o]=1,sumv[o]=0;
}
void build(int l,int r,int o){
if(l>r)return;
if(l==r) {
sumv[o]=1;
if(l==1)sumv[o]=0;
return;
}
int mid=(l+r)>>1;
build(l,mid,lson);
build(mid+1,r,rson);
pushup(o);
}
void modify(int l,int r,int o,int L,int R){
if(l>r||r<L||l>R)return;
if(lazy[o])return;
if(l>=L&&r<=R){
sumv[o]=0;
lazy[o]=1;
return;
}
int mid=(l+r)>>1;
modify(l,mid,lson,L,R);
modify(mid+1,r,rson,L,R);
pushup(o);
}
int query(int l,int r,int o,int L,int R){ if(lazy[o]||l>r||r<L||l>R)return 0;
if(l>=L&&r<=R)return sumv[o];
int mid=(l+r)>>1;
return query(l,mid,lson,L,R)+query(mid+1,r,rson,L,R);
}
void update(int a,int b){
while(top[a]!=top[b]){
if(dep[top[a]]>dep[top[b]]) swap(a,b);
modify(1,n,1,A[top[b]],A[b]),b=f[top[b]];
}
if(a==b)return ;
if(dep[a]>dep[b]) swap(a,b);
modify(1,n,1,A[son[a]],A[b]);
} int lookup(int a,int b){
int sums=0;
while(top[a]!=top[b]){
if(dep[top[a]]>dep[top[b]]) swap(a,b);
sums+=query(1,n,1,A[top[b]],A[b]);
b=f[top[b]];
}
if(a==b) return sums;
if(dep[a]>dep[b])swap(a,b);
sums+=query(1,n,1,A[son[a]],A[b]);
return sums;
} struct Edge{
int from,to,t;
Edge(int from=0,int to=0,int t=0):from(from),to(to),t(t){}
}edges[M];
struct Ask{
int a,b,c;
Ask(int a=0,int b=0,int c=0):a(a),b(b),c(c){}
}asks[Q];
typedef pair<int,int> P;
map<P,int>idx;
int curr;
int answer[Q];
int cnans;
int main(){
//setIO("input");
scanf("%d%d",&n,&m);
for(int i=1;i<=m;++i){
int a,b;
scanf("%d%d",&a,&b);
if(a>b)swap(a,b);
edges[i]=Edge(a,b,0);
idx[P(a,b)]=i;
}
while(1){
int a,b,c;
scanf("%d%d%d",&c,&b,&a);
if(c==-1) break;
if(a>b)swap(a,b);
asks[++curr]=Ask(a,b,c);
if(c!=1&&idx[P(a,b)]) edges[idx[P(a,b)]].t=1;
}
for(int i=1;i<=m;++i)
if(edges[i].t!=1)
addedge(edges[i].from,edges[i].to),addedge(edges[i].to,edges[i].from);
dfs1(1,0,0);
dfs2(1,1);
build(1,n,1);
for(int i=2;i<=cnt;i+=2)
if(!tag[i]){
tag[i]=tag[i^1]=1;
update(uu[i],vv[i]);
} for(int i=curr;i>=1;--i)
{
if(asks[i].c==1) answer[++cnans]=lookup(asks[i].a,asks[i].b);
else update(asks[i].a,asks[i].b);
}
for(int i=cnans;i>=1;--i) printf("%d\n",answer[i]);
return 0;
}

  

洛谷 P2542 [AHOI2005]航线规划 树链剖分_线段树_时光倒流_离线的更多相关文章

  1. 洛谷 P2542 [AHOI2005]航线规划 解题报告

    P2542 [AHOI2005]航线规划 题目描述 对Samuel星球的探险已经取得了非常巨大的成就,于是科学家们将目光投向了Samuel星球所在的星系--一个巨大的由千百万星球构成的Samuel星系 ...

  2. 洛谷P2542 [AHOI2005]航线规划(LCT,双连通分量,并查集)

    洛谷题目传送门 太弱了不会树剖,觉得LCT好写一些,就上LCT乱搞,当LCT维护双连通分量的练手题好了 正序删边是不好来维护连通性的,于是就像水管局长那样离线处理,逆序完成操作 显然,每个点可以代表一 ...

  3. 洛谷 P2542 [AHOI2005]航线规划(Link-cut-tree)

    题面 洛谷 bzoj 题解 离线处理+LCT 有点像星球大战 我们可以倒着做,断边变成连边 我们可以把边变成一个点 连边时,如果两个点本身不联通,就\(val\)赋为\(1\),并连接这条边 如果,两 ...

  4. 【洛谷5439】【XR-2】永恒(树链剖分,线段树)

    [洛谷5439][XR-2]永恒(树链剖分,线段树) 题面 洛谷 题解 首先两个点的\(LCP\)就是\(Trie\)树上的\(LCA\)的深度. 考虑一对点的贡献,如果这两个点不具有祖先关系,那么这 ...

  5. 【BZOJ5507】[GXOI/GZOI2019]旧词(树链剖分,线段树)

    [BZOJ5507][GXOI/GZOI2019]旧词(树链剖分,线段树) 题面 BZOJ 洛谷 题解 如果\(k=1\)就是链并裸题了... 其实\(k>1\)发现还是可以用类似链并的思想,这 ...

  6. bzoj 4034 [HAOI2015] T2(树链剖分,线段树)

    4034: [HAOI2015]T2 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1536  Solved: 508[Submit][Status] ...

  7. bzoj 1036 [ZJOI2008]树的统计Count(树链剖分,线段树)

    1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 10677  Solved: 4313[Submit ...

  8. poj 3237 Tree(树链剖分,线段树)

    Tree Time Limit: 5000MS   Memory Limit: 131072K Total Submissions: 7268   Accepted: 1969 Description ...

  9. bzoj 3626 [LNOI2014]LCA(离线处理+树链剖分,线段树)

    3626: [LNOI2014]LCA Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1272  Solved: 451[Submit][Status ...

随机推荐

  1. Linux就该这么学 20181002(第二章基础命令)

    参考链接https://www.linuxprobe.com/ 忘记密码操作 启动页面 默认按e 在linux16行后空格 rd.break ctrl + x mount -o remount,rw ...

  2. python 3.x 学习笔记9 (面向对象)

    1.面向对象 面向对象是一种对现实世界理解和抽象的方法,是计算机编程技术发展到一定阶段后的产物. 2.类(class): 一个类即是对一类拥有相同属性的对象的抽象.蓝图.原型.在类中定义了这些对象的都 ...

  3. MyBatis数据持久化(六)resultMap使用

    resultMap是MyBatis最强大也是最重要的特性,使用它我们可以將查询结果或者sql输入条件和一个HashMap的key进行映射,大大简化了sql查询条件的输入,而且使得数据的查询变得非常灵活 ...

  4. JavaScript学习——使用JS实现首页轮播图效果

    1.相关技术 获取元素 document.getElementById(“id 名称”) 事件(onload) 只能写一次并且放到body标签中 定时操作:setInterval(“changeImg ...

  5. Mysql数据库批量添加数据

    1.目的: 用于向Mysql表中批量插入数据 2.过程: 3.语句 #删除已有存储过程 DROP PROCEDURE if exists insertdata; #设定命令行出现//后再执行语句 de ...

  6. QT笔记 -- (3) 为QLabel添加鼠标响应方法1

    参考 http://qt-project.org/wiki/Make-a-QLabel-Clickable 1.首先重载QLabel的mousePressEvent,这样点击QLabel时就能发出cl ...

  7. FCC编程题之中级算法篇(下)

    介绍 本篇是"FCC编程题之中级算法篇"系列的最后一篇 这期完结后,下期开始写高级算法,每篇一题 目录 1. Smallest Common Multiple 2. Finders ...

  8. MySQL常用技能篇

    写在之前的话: 之前一直在用MSSERVER,刚用MySQL时有很多的不适应.就此小结一下工作中遇到的问题和场景(用的不是很深入,供初学者参考),文中出现的局限性欢迎指出 MySQL有客户端式(SQL ...

  9. 路飞学城Python-Day15(模块二思维导图)

  10. NodeJS学习笔记 (4)网络服务-http(ok)

    原文:https://github.com/chyingp/nodejs-learning-guide 自己敲代码: http模块概览 大多数nodejs开发者都是冲着开发web server的目的选 ...