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 ...
随机推荐
- Android 架构组件-Lifecycle、LiveData、ViewModel
Lifecycle Lifecycle组件包括LifecycleOwner.LifecleObserver,能方便监听Activity或者Fragment的生命周期. 步骤: 1.实现Lifecycl ...
- jdk1.7中hashmap扩容时不会产生死循环
在扩容时 transfer( ) 方法中 newTable 新数组 局部变量 table 旧数组 全局变量 当第一个链表进行while循环时 执行到 e.next = newTable[i]; 时 n ...
- 面试半年!三面阿里,四面蚂蚁金服,居然倒在了一个Java集合之Map上?
Map接口 Map与Collection并列存在.用于保存具有映射关系的数据:key-valueMap中的key和value都可以是任何引用类型的数据Map中的key用set来存放,不允许重复,即同一 ...
- 思维导图软件iMindMap幻灯片设置功能介绍
我们运用iMindMap演示来播放幻灯片时,有没想过,我怎么改动幻灯片的播放时长,怎么设置它的播放速度这些基本设置呢.下面,本文就告诉你,我们该去哪里修改这些iMindMap幻灯片设置: 我们打开iM ...
- Win搭建JAVA环境
一:下载JDK 下载链接:https://www.oracle.com/java/technologies/javase-downloads.html 选择你的系统环境进行下载 二:安装JDK 直接运 ...
- Yali 2019-8-15 test solution
T1. 送货 Description 物流公司要用m辆车派送n件货物.货物都包装成长方体,第i件的高度为hi,重量为wi.因为车很小,一辆车上的货物必须垒成一摞.又因为一些不可告人的原因,一辆车上货物 ...
- Jinja2语法自动补全配置
Jinja2语法自动补全配置 说明 在使用Pycharm社区版进行Web开发时,Jiaja2的语法是不会自动提示补全的,为了提高开发效率,需要根据个人习惯进行一些常用语法的自动补全配置,具体如下. 配 ...
- Leetcode 周赛#200 题解
1535 找出数组游戏的赢家 #模拟+优化 题目链接 题意 给你一个由 不同 整数组成的整数数组 arr 和一个整数 k(\(1\leq k\leq1e9\)) .每回合游戏都在数组的arr[0] 和 ...
- Java基础教程——Set
Set·无序,不重复 HashSet 特点:没有重复数据,数据不按存入的顺序输出. HashSet由Hash表结构支持.不支持set的迭代顺序,不保证顺序. 但是Hash表结构查询速度很快. 创建集合 ...
- 使用zabbix监控Jenkins
一.监控架构图 二.实现思路 在 Jenkins 上安装 Metrics 插件,使 Jenkins 暴露 metrics api: 编写python代码从api抓取数据,并将数据解析为zabbix可以 ...