How far away ? HDU - 2586 【LCA】【RMQ】【java】
题目大意:求树上任意两点距离。
思路:
dis[i]表示i到根的距离(手动选根),则u、v的距离=dis[u]+dis[v]-2*dis[lca(u,v)]。
lca:u~v的dfs序列区间里,深度最小的节点即为u、v的lca(最近公共祖先),RMQ把它找到。
难点:
何为dfs序:
就是树上dfs依次遍历的节点编号的序列。它的特点是回溯时会【再次】经过非叶节点,非叶节点在dfs序中出现多次,且dfs序列个数>节点个数。
为什么要用dfs序:
因为u、v的lca只出现在u、v的dfs序间。比如树1 3、1 2。dfs遍历的节点依次为1 3 1 2。3、2的lca为其中间的1。
RMQ的dp[i][j]的理解:
请把dp[][]看成一维数组,也就是只看第一维来理解这句话:i:dfs序为i,j:以i开始,长度为2^j的区间里depth最小的dfs序节点。
初始状态:依次储存dfs序(即dp[i][0]=i)。
状态转移:看看两个子区间的两个值,哪个depth更小,该区间就可以更新了。
实现:
lca实现:dfs序储存+depth[]+RMQ
dfs序实现:pos[](下标:节点编号,储存:dfs序)+ t[](下标:dfs序,储存:节点编号)(最后的dis[]查询要节点编号)
(pos[]既可以存节点的第一个dfs序也可以存最后一个dfs序,因为dfs序区间保证了包含lca,而那个depth最小的就是lca,dp会把它找到)
dep[]实现:下标dfs序,储存对应编号的depth
RMQ实现:dp[i][j]
dis[]实现:dfs会遍历每个节点一次或多次,dis[v]=0表示第一次,更新,dis[v]!=0表示v是回溯,跳过。
static IO io=new IO();
static int n,m;
private static final int MAXN = 41000; static class Edge{
int v,next,dis; public Edge(int v, int next, int dis) {
this.v = v;
this.next = next;
this.dis = dis;
}
} // 边编号
static int size;
// 双向加边一定开2倍,数组越界异常在hdu里显示wa
static Edge[]edges=new Edge[MAXN<<1];
static int[]head=new int[MAXN];
static int[]dis=new int[MAXN]; // dfs序,dfs完成后的意义是dfs序长度
static int tot;
// 这三个数组下标都是dfs序,不刷新,因为tot从0开始会依次覆盖
static int[]pos=new int[MAXN<<1];
static int[]dep=new int[MAXN<<1];
static int[]t=new int[MAXN<<1]; // dp不刷新,因为状态转移只会处理子区间,2的次方保证可以对半分,手动赋初值
static int[][]dp=new int[MAXN<<1][22]; public static void main(String[] args) {
int T=io.nextInt();
while (T-->0){
n=io.nextInt();m=io.nextInt(); Arrays.fill(head,-1);
Arrays.fill(dis,0);
size=0;
for (int i = 0; i < n - 1; i++) {
int a=io.nextInt(),b=io.nextInt(),c=io.nextInt();
edges[size]=new Edge(b,head[a],c);
head[a]=size++;
edges[size]=new Edge(a,head[b],c);
head[b]=size++;
} tot=0;
dfs(1,0); for (int i=0;i<tot;i++)dp[i][0]=i;
for (int j=1;(1<<j)<=tot;j++){
for (int i=1;i+(1<<j)-1<=tot;i++){
dp[i][j]= dep[dp[i][j-1]]<dep[dp[i+(1<<j-1)][j-1]]?
dp[i][j-1]:dp[i+(1<<j-1)][j-1];
}
} while (m-->0){
int a=io.nextInt(),b=io.nextInt();
// 因为dp[u][k]的u一定要左节点,dp[v-(1<<k)+1][k]的v一定要右节点
int u=Math.min(pos[a],pos[b]),v= Math.max(pos[a],pos[b]); int k=(int)(Math.log(v-u+1)/Math.log(2));
int dfsfa=dep[dp[u][k]]<dep[dp[v-(1<<k)+1][k]]?
dp[u][k]:dp[v-(1<<k)+1][k];
int fa=t[dfsfa];
io.println(dis[a]+dis[b]-2*dis[fa]);
}
}
} static void dfs(int u,int de){
pos[u]=tot;dep[tot]=de;t[tot++]=u;
for (int i=head[u];i!=-1;i=edges[i].next){
if (dis[edges[i].v]!=0)continue;
dis[edges[i].v]=dis[u]+edges[i].dis;
dfs(edges[i].v,de+1);
// 这一步就保证了lca出现在u、v的dfs序间
// 如树1 3、1 2,dfs序为1 3 1 2,这一步保证了1 再次出现在3 2间
dep[tot]=de;t[tot++]=u;
}
}
How far away ? HDU - 2586 【LCA】【RMQ】【java】的更多相关文章
- hdu 2586(裸LCA)
传送门 题意: 某村庄有n个小屋,n-1条道路连接着n个小屋(无环),求村庄A到村庄B的距离,要求是经过任一村庄不超过一次. 题解: 求出 lca = LCA(u,v) , 然后答案便是dist[u] ...
- How far away ? HDU - 2586
How far away ? Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)To ...
- HDU 2586(LCA欧拉序和st表)
什么是欧拉序,可以去这个大佬的博客(https://www.cnblogs.com/stxy-ferryman/p/7741970.html)巨详细 因为欧拉序中的两点之间,就是两点遍历的过程,所以只 ...
- 【10.10校内测试】【线段树维护第k小+删除】【lca+主席树维护前驱后驱】
贪心思想.将a排序后,对于每一个a,找到对应的删除m个后最小的b,每次更新答案即可. 如何删除才是合法并且最优的?首先,对于排了序的a,第$i$个那么之前就应该删除前$i-1$个a对应的b.剩下$m- ...
- 【LCA求最近公共祖先+vector构图】Distance Queries
Distance Queries 时间限制: 1 Sec 内存限制: 128 MB 题目描述 约翰的奶牛们拒绝跑他的马拉松,因为她们悠闲的生活不能承受他选择的长长的赛道.因此他决心找一条更合理的赛道 ...
- HDU - 2586 How far away ?(LCA模板题)
HDU - 2586 How far away ? Time Limit: 1000MS Memory Limit: 32768KB 64bit IO Format: %I64d & ...
- hdu 2586 How far away ?倍增LCA
hdu 2586 How far away ?倍增LCA 题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=2586 思路: 针对询问次数多的时候,采取倍增 ...
- LCA(最近公共祖先)--tarjan离线算法 hdu 2586
HDU 2586 How far away ? Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/ ...
- 【Java】代处理?代理模式 - 静态代理,动态代理
>不用代理 有时候,我希望在一些方法前后都打印一些日志,于是有了如下代码. 这是一个处理float类型加法的方法,我想在调用它前打印一下参数,调用后打印下计算结果.(至于为什么不直接用+号运算, ...
随机推荐
- MySql数据库在NodeJS中简单的基本操作
阅读目录 一:连接数据库 二:数据的增删改查操作 2.1 数据库新增和查询数据 2.2 获取该数据的主键值 2.3 多语句查询 回到顶部 一:连接数据库 const mysql = require(' ...
- volatile分析
volatile三大特性: 1. 内存可见性 2.不保证原子性 3. 禁止重排序内存屏障的概念:memory barrier是一个CPU指令.指令逻辑:a.确保一些特定操作执行顺序 b.影响一些数据的 ...
- C# — Windows服务安装后自动停止问题
今天在使用VS创建一个Windows服务时,为了得到一些提示,引用了Windows.Forms程序集,然后使用MessageBox.Show()方法渴望得到一些弹窗提示: 但是最后在安装好服务后,在任 ...
- 《通过C#学Proto.Actor模型》之Supervision
Supervision,字面意思是监督,是父Actor发现子Actor有异常发生后,对子Actor产用保种策略处理的机制,如果父Actor不处理,则往上传递. 子Actor发生异常后处理的策略有: R ...
- 全解史上最快的JOSN解析库 - alibaba Fastjson
JSON,全称:JavaScript Object Notation,作为一个常见的轻量级的数据交换格式,应该在一个程序员的开发生涯中是常接触的.简洁和清晰的层次结构使得 JSON 成为理想的数据交换 ...
- Linux 修改本地时间 (centos为例)
1. tzselect [root@xxxx etc]# tzselect --- 选择时区命令 Please identify a location so that time zone rules ...
- Openstack基础环境交换机常用配置(CISCO 3750G为例)
NOTE: 用户模式提示符为:cisco> 特权模式提示符为:cisco# 全局配置模式提示符为:cisco(config)# 端口模式提示符为:cisco(config-if)# 基础操作 ...
- Static Sushi AtCoder - 4118 (技巧枚举)
Problem Statement "Teishi-zushi", a Japanese restaurant, is a plain restaurant with only o ...
- Notepad++ PluginManager安装常用插件
参考: https://blog.csdn.net/u014611492/article/details/80849211
- java valueOf()函数
valueOf() 方法用于返回给定参数的原生 Number 对象值,参数可以是原生数据类型, String等. 该方法是静态方法.该方法可以接收两个参数一个是字符串,一个是基数. 语法 该方法有以下 ...