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 题解的更多相关文章
- 【题解】 Luogu P4312 / SP4155 [COCI 2009] OTOCI / 极地旅行社
原题地址:P4312 [COCI 2009] OTOCI / 极地旅行社/SP4155 OTOCI - OTOCI lct入门难度的题,十分弱智(小蒟蒻说lct是什么,能吃吗?) bridge操作判联 ...
- SPOJ - OTOCI LCT
OTOCI Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://acm.hust.edu.cn/vjudge/problem/viewProblem. ...
- [luogu]P4312 [COCI 2009] OTOCI / 极地旅行社(LCT)
P4312 [COCI 2009] OTOCI / 极地旅行社 题目描述 不久之前,Mirko建立了一个旅行社,名叫"极地之梦".这家旅行社在北极附近购买了N座冰岛,并且提供观光服 ...
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...
- noip2016十连测题解
以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...
- BZOJ-2561-最小生成树 题解(最小割)
2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ...
- Codeforces Round #353 (Div. 2) ABCDE 题解 python
Problems # Name A Infinite Sequence standard input/output 1 s, 256 MB x3509 B Restoring P ...
- 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解
题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...
- 2016ACM青岛区域赛题解
A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Jav ...
- poj1399 hoj1037 Direct Visibility 题解 (宽搜)
http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...
随机推荐
- XTTS系列之四:迷迷糊糊的并行度
项目测试组又反馈一个问题,XTTS执行全量备份速度慢,影响测试进度. 实际算了下,平均速度才150MB/s.. 这个速度在客户生产环境的确是不够看,首先询问是否开了并行,开了多少? 回复是说有开32个 ...
- Set_HashSet_TreeSet_小记
Set接口:Set集合继承自Collection集合 Set:底层数据结构是一个哈希表,能保证元素是唯一的,元素不重复!它通过它的子实现了HashSet集合去实例化,HashSet集合底层是HashM ...
- 将SQL从phpmyadmin导入到自己的Navicat
数据库名称太有指向性了 容易把自己给干没了 一码得码的一塌糊涂 就不放图了 进入你的phpmyadmin 注意一定要好好的进到数据库里面 点击导出 如果提示是 正在导出数据库"{你的数据库名 ...
- salesforce零基础学习(一百二十九)Lead Convertion 有趣的经历
本篇参考:https://help.salesforce.com/s/articleView?id=000382564&type=1 Lead Convertion 是salesforce中s ...
- React:如何在普通函数中使用Hook
解决方案
- Echarts: 同时显示柱状图和前端
完整代码 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF- ...
- 使用JDK自带工具调优JVM的常用命令
前言 对于Java进程常见问题,可以通过JVM监控工具(比如Prometheus).Arthas等,或者使用JDK自带的工具.如果第三方监控工具线上没有的话,对jdk自带的工具就要多熟悉熟悉. 线上J ...
- 从源码角度了解Vue生命周期
每个Vue应用都是通过new Vue()创建一个Vue实例开始.Vue()函数可以传入选项Options,常见的有el.template和data选项等. el 只在new创建实例时生效,其值可以是一 ...
- 玄 - 利用DLL通知回调函数注入shellcode(上)
序 偶然看到某国外大佬发布新技术-"Threadless"进程注入技术,据说可过EDR(确实可),总结该技术原理 - 在远程目标进程中利用DLL通知回调机制执行shellcode, ...
- 6、Mybatis之高级查询
6.1.创建接口.映射文件和测试类 ++++++++++++++++++++++++++分割线++++++++++++++++++++++++++ 注意namespace属性值为对应接口的全限定类名 ...