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. xmanager使用

    Xmanager全称Netsarang Xmanager,是国外一套非常优秀的远程监控软件.在UNIX/Linux和Windows网络环境中,Xmanager是最好的连通解决方案.我推 荐大家下载En ...

  2. Linux就该这么学 20181005(第八章防火墙)

    参考链接https://www.linuxprobe.com/ vim /etc/sysconfig/network-scripts/ifcfg-ens32 网络配置0 nmtui 网络配置1 nm- ...

  3. VM虚拟机上的CentOS 7系统重置root用户密码

    1.开机在进入CentOS系统时(还未进入系统内)的系统选择页面时按E键进入系统编辑模式 2.找到Linux16开头的这行代码,用方向键将光标移动至这行代码的结尾,键入一个空格和rd.break,然后 ...

  4. Objective-C —内存管理(上)

    内存管理 一.为什么要进行内存管理 移动设备的内存极其有限,每个app所能占用的内存是有限制的 下列行为都会增加一个app的内存占用 创建一个OC对象 定义一个变量 调用一个函数或者方法 内存占用多大 ...

  5. 从SQL注入谈数据访问层

    什么是SQL注入? SQL注入就是应用程序的开发人员未预期的吧SQL语句传入到应用程序的过程,如果直接使用用户输入的值来构建SQL语句的应用程序是很可能会受到SQL注入攻击的.特别是基于浏览器的网络应 ...

  6. Base64就是一种 基于64个可打印字符来表示二进制数据的表示方法

    Base64编码是从二进制到字符的过程. Base64编码主要用在传输.存储.表示二进制等领域,还可以用来加密,但是这种加密比较简单. byte[] byteArray = Encoding.UTF8 ...

  7. js闭包的用途详解

    js闭包可以用在许多地方.它的最大用处有两个,一个是前面提到的可以读取函数内部的变量,另一个就是让这些变量的值始终保持在内存中 我们来看看闭包的用途.事实上,通过使用闭包,我们可以做很多事情.比如模拟 ...

  8. bind方法使用案例

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  9. hiho1055/hdu1561 - 树形dp转换成背包

    题目链接 输入:一棵树,每个节点一个权值. 输出:包括1号节点在内的m个节点组成的连通分量的权值和的最大值 hdu1561和hiho1055一样,只是变换了下说法 /***************** ...

  10. shell-3.bash的基本功能:多命令顺序执行与管道符

    1. 2.