OTOCI

题目大意

给定 \(n\) 个带权的点,需要进行四种操作:查询两点连通性;加边;修改点权;查询两点路径的权值和。

思路分析

首先观察题目,我们会发现,在所有的操作结束后,所有的点构成一个森林,这是因为题目中的加边是建立在两点不连通的基础上的,所以不会形成任何的环,到最后自然形成了一个森林。

那么,我们可以不用理会题目的“要求在线处理所有操作”,使用离线大法跑树链剖分来解决了。

具体的说,我们先将所有的询问存下来离线处理,先用并查集维护两点的连通性和加边操作,然后在最后形成的森林上跑树剖,再依次处理每个询问。

时间复杂度:\(O(n\log^2n)\)。

代码

#include <bits/stdc++.h>
using namespace std;
const int N=100100;//双向边 int to[N],nxt[N],head[N],w[N];
int fat[N],son[N],dep[N],siz[N],top[N],dfn[N],rnk[N];//树剖七件套
int idx,n,m,in1,in2,q,cnt;
int fa[N],inp[N*3][3];char op[15];//并查集,询问 int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}//常规并查集
void add(int u,int v){idx++;to[idx]=v;nxt[idx]=head[u];head[u]=idx;}
void Swap(int &x,int &y){int t=x;x=y;y=t;} struct STn{int l,r,sum;};//线段树求区间和
struct ST{
STn a[N<<2];
void build(int p,int l,int r){
a[p].l=l;a[p].r=r;
if(a[p].l==a[p].r){a[p].sum=w[rnk[a[p].l]];return ;}
int mid=(a[p].l+a[p].r)>>1;
build(p<<1,l,mid);build(p<<1|1,mid+1,r);
a[p].sum=a[p<<1].sum+a[p<<1|1].sum;return ;
}
void change(int p,int x,int k){//单点修改
if(a[p].l==a[p].r){a[p].sum=k;return ;}
int mid=(a[p].l+a[p].r)>>1;
if(x<=mid) change(p<<1,x,k);else change(p<<1|1,x,k);
a[p].sum=a[p<<1].sum+a[p<<1|1].sum;return ;
}
int ask(int p,int l,int r){//求和
if(l<=a[p].l&&a[p].r<=r) return a[p].sum;
int mid=(a[p].l+a[p].r)>>1;
return ((l<=mid)?ask(p<<1,l,r):0)+((r>mid)?ask(p<<1|1,l,r):0);
}
}tree; void dfs_1(int s,int gr){//常规树剖dfs
dep[s]=dep[gr]+1;fat[s]=gr;
son[s]=-1;siz[s]=1;
for(int i=head[s];i;i=nxt[i]){
int v=to[i];
if(v==gr) continue;
dfs_1(v,s);
siz[s]+=siz[v];
if(son[s]==-1||siz[v]>siz[son[s]]) son[s]=v;
}
} void dfs_2(int s,int tp){
top[s]=tp;dfn[s]=++cnt;rnk[cnt]=s;
if(son[s]==-1) return ;
dfs_2(son[s],tp);
for(int i=head[s];i;i=nxt[i]){
int v=to[i];
if(v!=son[s]&&v!=fat[s]) dfs_2(v,v);
}
} int query(int x,int y){//树剖求链和
int res=0;
while(top[x]!=top[y]){
if(dep[top[x]]<dep[top[y]]) Swap(x,y);
res+=tree.ask(1,dfn[top[x]],dfn[x]);
x=fat[top[x]];
}
res+=tree.ask(1,min(dfn[x],dfn[y]),max(dfn[x],dfn[y]));
return res;
} int main(){
scanf("%d",&n);
for(int i=1;i<=n;i++) fa[i]=i;
for(int i=1;i<=n;i++) scanf("%d",&w[i]);
scanf("%d",&q);
for(int i=1;i<=q;i++){
scanf("%s%d%d",op+1,&in1,&in2);
if(op[1]=='b'){
if(find(in1)==find(in2)) inp[i][0]=9;
else{inp[i][0]=10;fa[find(in1)]=find(in2);add(in1,in2);add(in2,in1);}//加边
}
if(op[1]=='p'){inp[i][0]=1;inp[i][1]=in1;inp[i][2]=in2;}
if(op[1]=='e'){
if(find(in1)!=find(in2)) inp[i][0]=11;
else{inp[i][0]=2;inp[i][1]=in1;inp[i][2]=in2;}
}
}
for(int i=1;i<=n;i++)
if(!fat[i]){dfs_1(i,0);dfs_2(i,i);}//跑多次树剖
tree.build(1,1,n);
for(int i=1;i<=q;i++){//离线操作
if(inp[i][0]==9){puts("no");continue;}
if(inp[i][0]==10){puts("yes");continue;}
if(inp[i][0]==11){puts("impossible");continue;}
if(inp[i][0]==1){tree.change(1,dfn[inp[i][1]],inp[i][2]);}
if(inp[i][0]==2){cout<<query(inp[i][1],inp[i][2])<<'\n';}//回答询问
}
return 0;
}

OTOCI 题解的更多相关文章

  1. 【题解】 Luogu P4312 / SP4155 [COCI 2009] OTOCI / 极地旅行社

    原题地址:P4312 [COCI 2009] OTOCI / 极地旅行社/SP4155 OTOCI - OTOCI lct入门难度的题,十分弱智(小蒟蒻说lct是什么,能吃吗?) bridge操作判联 ...

  2. SPOJ - OTOCI LCT

    OTOCI Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://acm.hust.edu.cn/vjudge/problem/viewProblem. ...

  3. [luogu]P4312 [COCI 2009] OTOCI / 极地旅行社(LCT)

    P4312 [COCI 2009] OTOCI / 极地旅行社 题目描述 不久之前,Mirko建立了一个旅行社,名叫"极地之梦".这家旅行社在北极附近购买了N座冰岛,并且提供观光服 ...

  4. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  5. noip2016十连测题解

    以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...

  6. BZOJ-2561-最小生成树 题解(最小割)

    2561: 最小生成树(题解) Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1628  Solved: 786 传送门:http://www.lyd ...

  7. Codeforces Round #353 (Div. 2) ABCDE 题解 python

    Problems     # Name     A Infinite Sequence standard input/output 1 s, 256 MB    x3509 B Restoring P ...

  8. 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解

    题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...

  9. 2016ACM青岛区域赛题解

    A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Jav ...

  10. poj1399 hoj1037 Direct Visibility 题解 (宽搜)

    http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...

随机推荐

  1. decode php解密代码,方便好用,请收藏

    <?php //已经加密的文件内容 $a = "eval(gzinflate(base64_decode('tVRNb+IwEL3vr/AhwomU5WOPVHSF2lSg7QJK0j ...

  2. 国标GB28181协议客户端开发(四)实时视频数据传输

    国标GB28181协议客户端开发(四)实时视频数据传输 本文是<国标GB28181协议设备端开发>系列的第四篇,介绍了实时视频数据传输的过程.通过解读INVITE报文中的SDP信息,读取和 ...

  3. python笔记:第二章基本数据类型

    基本数据类型 Python中的数据类型可以分为五大类:字符串.数字.容器.布尔.None 1.字符串 可以使用单引号或双引号创建字符串,可以用加号将两个字符串合并 name = '小明' age = ...

  4. 跟着 GPT-4 从0到1学习 Golang 并发机制(三)

    目录 一.前言 二.开聊 2.1 关于 goroutine 泄露问题 2.2 内存模型 2.3 Race Detector 检测数据竞争 三.总结 一.前言 话接上回<跟着 GPT-4 从0到1 ...

  5. 飞桨paddlespeech语音唤醒推理C定点实现

    前面的文章(飞桨paddlespeech语音唤醒推理C浮点实现)讲了飞桨paddlespeech语音唤醒推理的C浮点实现.但是嵌入式设备通常CPU频率低和memory小,在嵌入式设备上要想流畅的运行语 ...

  6. KVM 硬盘分区扩容(GPT与MBR两种分区、fdisk 与 growpart两种方法)

    因为认知顺序的原因,之前我都是用fdisk命令手工删除分区表后重建进行扩容,后面才发现可以用growpart命令. 实战建议直接点 AWS EC2 存储空间扩容 跳转过去参考,学习操作可以继续往下看. ...

  7. 2021-11-17 WPF初识

    StackPanel容器:默认竖直排列,Orientation="Horizontal"横向排列,超过就不会显示 wrapPanel:超过会自动换行 设置样式: <Windo ...

  8. Win10安装Oracle-21C

    1.前期工作 下载安装包:OracleXE213_Win64.zip 解压安装包 2.开始安装 注意:以管理员身份运行 ++++++++++++++++++++++分割线+++++++++++++++ ...

  9. AgileConfig-1.7.0 发布,支持 SSO 🎉🎉🎉

    AgileConfig 已经好久好久没有更新过比较大的功能了.一是 AgileConfig 本身的定位就是比较轻量,不想集成太多的功能.二是比较忙(懒).但是本次升级给大家带来了一个比较有用的功能 S ...

  10. 《SQL与数据库基础》22. 分库分表(二)

    目录 分库分表(二) MyCat分片规则 范围分片 取模分片 一致性hash分片 枚举分片 应用指定算法 固定分片hash算法 字符串hash解析 按天分片 自然月分片 MyCat原理 MyCat管理 ...