<题目链接>

题目大意:
给定一颗带点权的树,进行两种操作,k=0,更改某一点的点权,k!=0,输出a~b路径之间权值第k大的点的点权。

解题分析:
先通过RMQ的初始化,预处理pre[]数组,并且求出a和b的LCA。然后利用LCA将a、b路径上所有点的点权全部存储起来,将其排序,就可得到a、b路径上权值第k大的点权。具体操作为:利用pre[]数组,从a到LCA的所有点权全部相加,然后再将从b到LCA的所有点权相加,就可以得到a、b路径上所有点的点权。下面用的是LCA转RMQ的方法求解。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std; const int maxn = 8e4+;
struct Edge{
int to,next;
}edge[maxn<<];
int n,q,cnt,num,tot,pre[maxn],val[maxn],fa[maxn];
int st[maxn<<][],ver[maxn<<],dep[maxn<<],first[maxn],path[maxn]; bool cmp(int a,int b){ return a > b;}
int Min(int l,int r){
return dep[l]<dep[r]?l:r;
}
void addedge(int u,int v){
edge[num].to = v,edge[num].next = pre[u];
pre[u] = num++;
}
void dfs(int u,int deep){
ver[++cnt] = u; first[u] = cnt; dep[cnt] = deep;
for(int i = pre[u]; i != -; i = edge[i].next){
int v = edge[i].to;
if(fa[u] == v) continue;
fa[v] = u;
dfs(v,deep+);
ver[++cnt] = u; dep[cnt] = deep;
}
}
void RMQ_init(){
for(int i = ; i <= cnt; i++)
st[i][] = i;
for(int j = ; ( << j) <= cnt; j++)
for(int i = ; i + ( << j) < cnt; i++)
st[i][j] = Min(st[i][j-],st[i+(<<(j-))][j-]);
}
int findLCA(int l,int r){
int k = (int)(log(r - l + 1.0) / log(2.0));
return ver[Min(st[l][k],st[r-(<<k)+][k])];
}
//以上是LCA转RMQ的模板
int main(){
int k,a,b;
while(scanf("%d%d",&n,&q)!=EOF){
cnt = num = ;
memset(pre,-,sizeof(pre));
for(int i = ; i <= n; i++)
scanf("%d",&val[i]);
for(int i = ; i < n; i++){
scanf("%d%d",&a,&b);
addedge(a,b);
addedge(b,a);
}
dfs(,);
RMQ_init();
while(q--){
scanf("%d%d%d",&k,&a,&b);
if(k == )val[a] = b;
else{
int x = first[a], y = first[b];
if(x > y) swap(x,y);
int lca = findLCA(x,y); //得到最近公共祖先
tot = ;
while(a != lca){ //将a到最近公共祖先的路劲上所有的点权放入path数组
path[++tot] = val[a];
a = fa[a];
}
while(b != lca){ //将a到最近公共祖先的路劲上所有的点权放入path数组
path[++tot] = val[b];
b = fa[b];
}
path[++tot] = val[lca]; //再加上lca的点权
if(k > tot) printf("invalid request!\n");
else{
sort(path+,path++tot,cmp); //将a->b路劲上所有点权排序,然后就可以得到第k大的点权
printf("%d\n",path[k]);
}
}
}
}
return ;
}

2018-10-23

hdu 3078 Network (暴力)+【LCA】的更多相关文章

  1. HDU - 3078 Network(暴力+LCA)

    题目大意:给出n个点的权值.m条边,2种操作 0 u num,将第u个点的权值改成num k u v,询问u到v这条路上第k大的权值点 解题思路:该点的话直接该,找第k大的话直接暴力 #include ...

  2. HDU 3078 Network(LCA dfs)

    Network [题目链接]Network [题目类型]LCA dfs &题意: 给出n个点的权值,m条边,2种操作 0 u num,将第u个点的权值改成num k u v,询问u到v这条路上 ...

  3. HDU 3078 Network LCA

    题意:n个点 m个询问,下面一行是n 个点的权值 再下面n-1行是双向的边 然后m个询问:k u v 若k==0,则把u点的权值改为v,否则回答u->v之间最短路经过点的权值中  第k大的值是多 ...

  4. HDU 3078 Network

    简单的  RMQ:  先预处理得到  所有 节点的 公共祖先  和  dfs 得到所有节点的父亲节点:  然后  询问时,从自己出发向上找父亲, 然后  得到所有的节点:排序一下 不知道  这题这样也 ...

  5. HDU 2460 Network(双连通+树链剖分+线段树)

    HDU 2460 Network 题目链接 题意:给定一个无向图,问每次增加一条边,问个图中还剩多少桥 思路:先双连通缩点,然后形成一棵树,每次增加一条边,相当于询问这两点路径上有多少条边,这个用树链 ...

  6. poj 3417 Network(tarjan lca)

    poj 3417 Network(tarjan lca) 先给出一棵无根树,然后下面再给出m条边,把这m条边连上,然后每次你能毁掉两条边,规定一条是树边,一条是新边,问有多少种方案能使树断裂. 我们设 ...

  7. HDU 3078:Network(LCA之tarjan)

    http://acm.hdu.edu.cn/showproblem.php?pid=3078 题意:给出n个点n-1条边m个询问,每个点有个权值,询问中有k,u,v,当k = 0的情况是将u的权值修改 ...

  8. HDU 3078 (LCA+树链第K大)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3078 题目大意:定点修改.查询树中任意一条树链上,第K大值. 解题思路: 先用离线Tarjan把每个 ...

  9. HDU 2460 Network(桥+LCA)

    http://acm.hdu.edu.cn/showproblem.php?pid=2460 题意:给出图,求每次增加一条边后图中桥的数量. 思路: 先用tarjan算法找出图中所有的桥,如果lowv ...

随机推荐

  1. jmeter测试mysql遇到的问题

    1. 1.防火墙未关 错误: Communications link failure The last packet sent successfully to the server was 0 mil ...

  2. 通过cmd 使用 InstallUtil.exe 命令 操作 windows服务 Windows Service

    要安装windows service 首先要找到 InstallUtil.exe,InstallUtil.exe位置在 C:\Windows\Microsoft.NET\Framework\v4.0. ...

  3. Spring如何使用JdbcTemplate调用存储过程的三种情况

    注:原文 <Spring如何使用JdbcTemplate调用存储过程的三种情况 > Spring的SimpleJdbcTemplate将存储过程的调用进行了良好的封装,下面列出使用Jdbc ...

  4. Confluence 6 从其他备份中恢复数据

    一般来说,Confluence 数据库可以从 Administration Console 或者 Confluence Setup Wizard 中进行恢复. 如果你在恢复压缩的 XML 备份的时候遇 ...

  5. kafka架构浅显理解

    Kafka的概念: 1. AMQP协议 Advanced Message Queuing Protocol (高级消息队列协议) The Advanced Message Queuing Protoc ...

  6. 【mysql】编码问题

    原始数据是unicode,存入数据库.需要注意的几个地方: 1.建立数据库时,选择编码方式为utf8 -- UTF-8 Unicode 2.代码中建立数据库连接时,选择charset=utf8 3.存 ...

  7. Python基础知识之大杂烩

    一.range 和 xrange 的区别 xrange 与 range 基本上都是在循环的时候用,两者的用法完全相同.所不同的是xrange生成的是一个生成器,而range生成的是一个list对象. ...

  8. Practical Web Penettation Testing (the first one Mutillidae 大黄蜂 之二)

    1.how  to use dpkg cmmand first it can be used for list all software , dpkg -l  (由于kali linux 没有启动所以 ...

  9. 安装AngularJS Batarang遇到的问题

    AngularJS Batarang是AngularJS在谷歌浏览器上的一个调试工具,因为国内目前无法访问谷歌浏览器应用商店,所以Batarang只能离线安装.不过在安装这个插件的过程中遇到了一些麻烦 ...

  10. 第三周学习总结-Java

    2018年7月29日 这是暑假第三周.这一周我把找到的Java教学视频看完了. 本周学到了Java剩余的基础知识,比如:抽象类.接口.内部类.几种常用类.IO流.多态.多线程等等. 因为没有书,所以我 ...