CF1416D Graph and Queries
本题解用于作者加深算法印象,也欢迎各位的阅读。
题目大意
给你一张无向图,并给你两种操作:
\(1~v\) :找到当前点 \(v\) 所在的联通块内权值最大的点,输出该点权值并将其权值改为 \(0\) 。
\(2~i\) :删去编号为 \(i\) 的边。
题解
然后你发现这个东西不是很好搞,但是这里提供了一个很好的维护联通块的东西——\(Kruskal\) 重构树。
我们可以将每条边删去的时间戳记为这条边的权值,可以轻易的发现,如果我们将边权从大到小(即在时间上从后往前)建起 \(Kruskal\) 重构树的话,每个子树都代表着一个时间点上的联通块,我们只需要对这个子树做我们的 \(1\) 操作就可以了,这个可以用 \(dfs\) 序加线段树来实现,这里就不多赘述了。
以上。
代码如下:
#include<bits/stdc++.h>
using namespace std;
const int N=2e5+5,M=3e5+5,Q=5e5+5;
int n,m,q,a[N<<1],b[N<<1];
struct Edge{int u,v,w,tag;}e[M];
bool cmp(Edge a,Edge b){return a.tag>b.tag;}
struct operation{int opt,x;}s[Q];
struct Seg_Tree
{
struct Node{int data,pos;}tr[N<<4];
void up(int u)
{
if(tr[u<<1].data>tr[u<<1|1].data) tr[u]=tr[u<<1];
else tr[u]=tr[u<<1|1];
}
void build(int u,int l,int r,int a[])
{
if(l==r){tr[u].data=a[l],tr[u].pos=l;return;}
int mid=(l+r)>>1;
build(u<<1,l,mid,a),build(u<<1|1,mid+1,r,a);
up(u);
}
int query(int u,int l,int r,int x,int y)
{
if(x<=l&&r<=y) return u;
int mid=(l+r)>>1,res=0,tmp;
if(x<=mid)
{
tmp=query(u<<1,l,mid,x,y);
if(tr[res].data<=tr[tmp].data) res=tmp;
}
if(y>mid)
{
tmp=query(u<<1|1,mid+1,r,x,y);
if(tr[res].data<=tr[tmp].data) res=tmp;
}
return res;
}
void del(int u,int l,int r,int x)
{
if(x<=l&&r<=x){tr[u].data=0;return;}
int mid=(l+r)>>1;
if(x<=mid) del(u<<1,l,mid,x);
else del(u<<1|1,mid+1,r,x);
up(u);
}
}Seg;
struct Kruskal_Tree
{
int size;
struct DSU
{
int fa[N<<1];
void init(){for(int i=1;i<(N<<1);++i)fa[i]=i;}
int find(int x){if(fa[x]!=x)fa[x]=find(fa[x]);return fa[x];}
}d;
struct Edge{int nxt,to;}e[N<<1];
int fir[N<<1],e_size;
void add(int u,int v){e[++e_size]=Edge{fir[u],v},fir[u]=e_size;}
struct Node{int fa[22],dep,data,mp;}tr[N<<1];
int dfn[N<<1],l[N<<1],r[N<<1],dfn_num;
void dfs(int u)
{
dfn[++dfn_num]=u,tr[u].mp=dfn_num;
l[u]=dfn_num;
// printf("%d %d\n",u,tr[u].data);
for(int i=fir[u];i;i=e[i].nxt)
{
tr[e[i].to].fa[0]=u;
tr[e[i].to].dep=tr[u].dep+1;
dfs(e[i].to);
}
r[u]=dfn_num;
}
int lca(int u,int v)
{
if(tr[u].dep<tr[v].dep) swap(u,v);
for(int i=20;i>=0;--i)
{
if(tr[tr[u].fa[i]].dep>=tr[v].dep)
u=tr[u].fa[i];
}
if(u==v) return u;
for(int i=20;i>=0;--i)
{
if(tr[u].fa[i]!=tr[v].fa[i])
u=tr[u].fa[i],v=tr[v].fa[i];
}
return tr[u].fa[0];
}
}t;
int main()
{
cin>>n>>m>>q;
t.size=n,t.d.init();
for(int i=1;i<=n;++i) scanf("%d",&a[i]);
for(int i=1;i<=m;++i) scanf("%d%d",&e[i].u,&e[i].v),e[i].tag=q+1;
// printf("\n----------------\n");
for(int i=1;i<=q;++i)
{
scanf("%d%d",&s[i].opt,&s[i].x);
if(s[i].opt==2) e[s[i].x].tag=i;
}
sort(e+1,e+1+m,cmp);
for(int i=1;i<=m;++i)
{
int fu=t.d.find(e[i].u);
int fv=t.d.find(e[i].v);
if(fu==fv) continue;
t.tr[++t.size].data=e[i].tag;
t.add(t.size,fu),t.add(t.size,fv);
// printf("%d %d %d\n",t.size,fu,fv);
t.d.fa[fu]=t.d.fa[fv]=t.size;
}
for(int i=1;i<=t.size;++i)
{
int tmp=t.d.find(i);
if(t.tr[tmp].mp) continue;
t.tr[tmp].dep=1;
t.dfs(tmp);
}
// for(int i=1;i<=t.size;++i)
// {
// printf("%d %d %d\n",t.tr[i].fa[0],t.tr[i].dep,t.tr[i].data);
// }
for(int i=1;i<=20;++i)
{
for(int j=1;j<=t.size;++j)
t.tr[j].fa[i]=t.tr[t.tr[j].fa[i-1]].fa[i-1];
}
for(int i=1;i<=t.size;++i) b[i]=a[t.dfn[i]];
Seg.build(1,1,t.size,b);
for(int i=1;i<=q;++i)
{
if(s[i].opt==2) continue;
int tmp=s[i].x;
for(int j=20;j>=0;--j)
{
if(t.tr[t.tr[tmp].fa[j]].data>i)
tmp=t.tr[tmp].fa[j];
}
tmp=Seg.query(1,1,t.size,t.l[tmp],t.r[tmp]);
// printf("%d %d ",tmp,Seg.tr[tmp].pos);
printf("%d\n",Seg.tr[tmp].data);
Seg.del(1,1,t.size,Seg.tr[tmp].pos);
}
}
CF1416D Graph and Queries的更多相关文章
- HDU 3726 Graph and Queries 平衡树+前向星+并查集+离线操作+逆向思维 数据结构大综合题
Graph and Queries Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Other ...
- [la P5031&hdu P3726] Graph and Queries
[la P5031&hdu P3726] Graph and Queries Time Limit: 10000/5000 MS (Java/Others) Memory Limit: ...
- HDU 3726 Graph and Queries (离线处理+splay tree)
Graph and Queries Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Other ...
- HDU 3726 Graph and Queries treap树
题目来源:HDU 3726 Graph and Queries 题意:见白书 思路:刚学treap 參考白皮书 #include <cstdio> #include <cstring ...
- HDU 3726 Graph and Queries(平衡二叉树)(2010 Asia Tianjin Regional Contest)
Description You are given an undirected graph with N vertexes and M edges. Every vertex in this grap ...
- UVALive5031 Graph and Queries(Treap)
反向操作,先求出最终状态,再反向操作. 然后就是Treap 的合并,求第K大值. #include<cstdio> #include<iostream> #include< ...
- UVa 1479 (Treap 名次树) Graph and Queries
这题写起来真累.. 名次树就是多了一个附加信息记录以该节点为根的树的总结点的个数,由于BST的性质再根据这个附加信息,我们可以很容易找到这棵树中第k大的值是多少. 所以在这道题中用一棵名次树来维护一个 ...
- uvalive 5031 Graph and Queries 名次树+Treap
题意:给你个点m条边的无向图,每个节点都有一个整数权值.你的任务是执行一系列操作.操作分为3种... 思路:本题一点要逆向来做,正向每次如果删边,复杂度太高.逆向到一定顺序的时候添加一条边更容易.详见 ...
- 【HDOJ】3726 Graph and Queries
Treap的基础题目,Treap是个挺不错的数据结构. /* */ #include <iostream> #include <string> #include <map ...
随机推荐
- c++中的几种函数调用约定(转)
C++中的函数调用约定(调用惯例)主要针对三个问题: 1.参数传递的方式(是否采用寄存器传递参数.采用哪个寄存器传递参数.参数压桟的顺序等): 参数的传递方式,最常见的是通过栈传递.函数的调用方将参数 ...
- Python可视化界面
可视化界面程序,本来不想写,只在console台运行就好,但是后来很多小伙伴都有这样的需求: 需要从redis中删除某个key的value,然后需要跟key去查,有些小伙伴不会用redis,就产生如下 ...
- 【Java从入门到精通】day08-包机制-JavaDoc生成文档
1.包机制 为了更好地组织类,Java提供了包机制,用于区别类名的命名空间. 包语句的语法格式为: package pkg1[.pkg2[.pkg3...]]; 一般利用公司域名倒置作为包名(如www ...
- 建议收藏,从零开始创建一个Activiti工作流,手把手教你完成
环境配置 项目环境: JDK1.8 tomcat7 maven3.5 开发工具: IDEA activiti7 创建项目 目标:创建一个maven项目,集成Activiti,并自动生成25张数据库表 ...
- Linux标准输入、标准输出、错误输出
Linux中的输入文件.输出文件.错误输出 文件名称 文件描述符 标准输入 0 (默认是键盘) 标准输出 1 (默认是屏幕) 标准错误 2 (默认是屏幕) 输出重定向 Linux中&表示后台运 ...
- SAP PI接口ESR IA配置,几种常用的 XSL 转换文档模板
在PI开发配置中字段映射一般分为Message Mapping(MM)和Imported Archives(IA)这两种形式.MM这种拉线的形式虽然看似方便,但是当接口更新和传输时往往比较麻烦,同时无 ...
- MindManager思维导图应用到办公中需要注意什么
MindManager思维导图是一个易于使用的项目管理软件,能很好地提高项目组的工作效率和小组成员之间的协作性.接下来,小编就为大家介绍三个能利用好该思维导图软件办公的技巧. 一.审阅会议--合作办公 ...
- CDR排钻教程-CorelDRAW服装设计中的排钻技术
服装设计一直都是一个很火热的行业,也是一个比较高端的行业,随着时代的步伐,以前的人都是用手绘的方式来设计服装,现在不一样了,电脑可以说普及到了每一个家庭,让软件以更快的速度,更准确的数据来设计服装中的 ...
- eNSP VLAN划分基础配置及Access接口
交换机内实现VLAN通信拓扑图: 一.修改主机名并保存 1.进入系统视图模式(配置模式) <Huawei>system-view 2.sysname命令修改主机名为yanyuda [Hua ...
- Guava中EventBus分析
EventBus 1. 什么是EventBus 总线(Bus)一般指计算机各种功能部件之间传送信息的公共通信干线,而EventBus则是事件源(publisher)向订阅方(subscriber)发送 ...