题解 UVA1479 【Graph and Queries】
\]
算是一道思维难度稍易,代码难度稍难的题吧。
\]
给出一张 \(n\) 个点,\(m\) 条边的图,点带权。需要支持三个操作:
D x
删掉编号为 \(x\) 的边Q x k
查询与节点 \(x\) 联通的所有节点中,点权第 \(k\) 大节点的点权C x v
将节点 \(x\) 点权改为 \(v\)
多组数据,每组数据最终需要输出所有查询的平均值 ( 保留 6 位 ) ,没有强制在线。
\]
不知道大家有没有做过 这道题 ,推荐先去做一下。
\(~\)
首先,对于同一个连通块里的所有节点,查询与任意一个节点连通的所有节点的第 \(k\) 大,都是查询该连通块里所有节点的第 \(k\) 大。已经很明显可以用并查集维护每个连通块的代表节点,再在这个代表节点上用一个数据结构维护连通块信息,支持合并,查询第 \(k\) 大。
我们发现权值线段树可以做到上述操作,尝试用权值线段树维护,每个节点开一个权值线段树。
\(~\)
对于操作 Q x k
\(:\)
权值线段树基本操作。
对于操作 C x v
\(:\)
我们可以看作是在 \(x\) 这个位置上少了一个原来的点权,再多了一个新的点权,两次插入操作即可解决。
对于操作 D x
\(:\)
\(......\) ,我们发现删掉一条边,不能有效使得一个连通块分裂成两个连通块,并且维护权值线段树。
\(~\)
注意到此题 没有强制在线 ,意味着,我们可以离线地把所有操作都读进来,然后去反着考虑这些询问。
这样一来,D x
操作就可以变为 \(:\) 加入一条编号为 \(x\) 的边。其余的两个操作不变。
我们发现添加一条边很容易维护 \(:\) 找出 \(u,v\) 所在的连通块 \(p,q\) ,若 \(p=q\) ,则无需操作;否则合并权值线段树 \(p\) 和权值线段树 \(q\) ,然后令 \(fa[q]=p\) 。
综上所述,我们就可以用 权值线段树 \(+\) 并查集 解决本题了。\((\) 当然什么 \(splay\),\(treap\) 启发式合并也行 \()\)
时空复杂度 \(\text{O(n log n)}\) 。
\]
#include<cstdio>
#include<cstring>
#define RI register int
using namespace std;
inline int read()
{
int x=0,f=1;char s=getchar();
while(s<'0'||s>'9'){if(s=='-')f=-f;s=getchar();}
while(s>='0'&&s<='9'){x=x*10+s-'0';s=getchar();}
return x*f;
}
const int N=6000100,M=6000100,Q=6000100,MLOGN=50000000;
const int INF=1e6;
int T;
int n,m,q;
int cnt;
long double ans;
int val[N];
struct Edge{
int u;
int v;
bool del;
}e[M];
char opt[Q];
int x[Q],k[Q];
int fa[N];
int get(int x)
{
if(fa[x]==x)return x;
return fa[x]=get(fa[x]);
}
int tot,root[N];
struct SegmentTree{
int lc,rc;
int cnt;
}t[MLOGN];
int New()
{
tot++;
t[tot].lc=t[tot].rc=t[tot].cnt=0;
return tot;
}
void insert(int &p,int l,int r,int delta,int val)
{
if(!p)
p=New();
t[p].cnt+=val;
if(l==r)return;
int mid=(l+r)>>1;
if(delta<=mid)
insert(t[p].lc,l,mid,delta,val);
else
insert(t[p].rc,mid+1,r,delta,val);
}
int merge(int p,int q)
{
if(!p||!q)
return p^q;
t[p].cnt+=t[q].cnt;
t[p].lc=merge(t[p].lc,t[q].lc);
t[p].rc=merge(t[p].rc,t[q].rc);
return p;
}
int ask(int p,int l,int r,int k)
{
if(l==r)
return l;
int mid=(l+r)>>1;
int rcnt=t[t[p].rc].cnt;
if(k<=rcnt)
return ask(t[p].rc,mid+1,r,k);
else
return ask(t[p].lc,l,mid,k-rcnt);
}
void link(int u,int v)
{
u=get(u),v=get(v);
if(u==v)return;
root[u]=merge(root[u],root[v]);
fa[v]=u;
}
void work()
{
tot=cnt=ans=q=0;
memset(root,0,sizeof(root));
for(RI i=1;i<=n;i++)
val[i]=read();
for(RI i=1;i<=m;i++)
e[i].u=read(),e[i].v=read();
char tmp[2];
while(scanf("%s",tmp),tmp[0]!='E')
{
opt[++q]=tmp[0];
switch(tmp[0])
{
case 'D':{
x[q]=read();
e[x[q]].del=true;
break;
}
case 'Q':{
x[q]=read(),k[q]=read();
cnt++;
break;
}
case 'C':{
x[q]=read(),k[q]=val[x[q]],val[x[q]]=read();
break;
}
}
}
for(RI i=1;i<=n;i++)
fa[i]=i,insert(root[i],-INF,INF,val[i],1);
for(RI i=1;i<=m;i++)
{
if(e[i].del)continue;
link(e[i].u,e[i].v);
}
for(RI i=q;i>=1;i--)
switch(opt[i])
{
case 'D':{
e[x[i]].del=false;
link(e[x[i]].u,e[x[i]].v);
break;
}
case 'Q':{
int p=get(x[i]);
int A=ask(root[p],-INF,INF,k[i]);
if(A==-INF||A==INF)
continue;
ans+=(long double)A/cnt;
break;
}
case 'C':{
int p=get(x[i]);
insert(root[p],-INF,INF,val[x[i]],-1);
val[x[i]]=k[i];
insert(root[p],-INF,INF,val[x[i]],1);
break;
}
}
printf("Case %d: %Lf\n",++T,ans);
}
int main()
{
while(n=read(),m=read(),n&&m) work();
return 0;
}
\]
题解 UVA1479 【Graph and Queries】的更多相关文章
- UVA1479 Graph and Queries
思路 恶心人的题目 还是类似永无乡一题的Treap启发式合并思路 但是由于加边变成了删边 所以应该离线后倒序处理 数组要开够 代码 #include <cstdio> #include & ...
- 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 ...
- CF1416D Graph and Queries
本题解用于作者加深算法印象,也欢迎各位的阅读. 题目大意 给你一张无向图,并给你两种操作: \(1~v\) :找到当前点 \(v\) 所在的联通块内权值最大的点,输出该点权值并将其权值改为 \(0\) ...
- UVALive5031 Graph and Queries(Treap)
反向操作,先求出最终状态,再反向操作. 然后就是Treap 的合并,求第K大值. #include<cstdio> #include<iostream> #include< ...
- UVa 1479 (Treap 名次树) Graph and Queries
这题写起来真累.. 名次树就是多了一个附加信息记录以该节点为根的树的总结点的个数,由于BST的性质再根据这个附加信息,我们可以很容易找到这棵树中第k大的值是多少. 所以在这道题中用一棵名次树来维护一个 ...
随机推荐
- linux下安装cmake方法(2)---直接用命令安装
1.linux环境下打开网页,输入上网账号密码,确保已经联网 2.打开终端:输入cmake --version,如果出现版本号,表明已经安装,如果显示没有安装cmake,则需要安装 3.在终端里输入: ...
- 快速回顾MySQL:简单查询操作
利用空闲时间花几分钟回顾一下 7.1 检索数据 为了查询出数据库表中的行(数据),使用SELECE语句. 格式: # 第一种 SELECT * FROM <table_name>; # 第 ...
- 有哪些让人相见恨晚的Python库(一)
对于我这个经常用python倒腾数据的人来说,下面这个库是真·相见恨晚 记得有一次我在服务器上处理数据时,为了解决Pandas读取超过2000W条数据就内存爆炸的问题,整整用了两天时间来优化.最后通过 ...
- Webpack实战(五):轻松读懂Webpack如何分离样式文件
在上一篇文章中我给大家分享了预处理器(loader),里面讲到了style-loader 和css-loader,有关样式引入的问题,但是上面的样式文件只是引入到style标签里面,并不是我想要的样式 ...
- Ansible Playbooks常用模块
File模块 在目标主机创建文件或目录,并赋予其系统权限 - name: create a file file:'path=/oot/foo.txt state=touch mode=0755 own ...
- wtforms 钩子函数
参考: https://www.cnblogs.com/wupeiqi/articles/8202357.html class LoginForm(Form): name = simple.Strin ...
- Sigmoid非线性激活函数,FM调频,胆机,HDR的意义
前几天家里买了个二手车子,较老,发现只有FM收音机,但音响效果不错,车子带蓝牙转FM,可以手机蓝牙播放音乐,但经过几次转换以及对FM的质疑,所以怀疑音质是否会剧烈下降,抱着试试的态度放了一个手机上的音 ...
- Egret学习-初次创建项目
最近无聊,好久没有写游戏了,决定学习下egret,主要原因:egret是h5框架,相比android和iPhone或cocos2dx来说不需要安装可以直接运行. 下面进入正题,开始学习egret 简单 ...
- 解决android sdk无法更新 更新慢的问题
使用不同平台开发android应用的时候都要先搭建开发环境. 这里介绍一下搭建开发环境过程中更新和下载android sdk的一种方法: 第一步:配置android sdk manager的代理服务, ...
- 深入浅出| java中的clone方法
每天进步一丢丢,连接梦与想 我们还年轻,但这不是你浪费青春的理由 克隆和复制 clone,有人称之为克隆,有人称之为复制,其实都是同一个东西 本文称之为"克隆",毕竟人家方法名叫& ...