hdu6115 Factory (LCA + 倍增)
Factory
Time Limit: 20000/10000 MS (Java/Others) Memory Limit: 132768/132768 K (Java/Others)
Total Submission(s): 399 Accepted Submission(s): 138
Problem Description
现在百度陆续开了许许多多的子公司。每家子公司又会在各城市中不断兴建属于该子公司的办公室。
由于各个子公司之间经常有资源的流动,所以公司员工常常想知道,两家子公司间的最小距离。
我们可以把子公司看成一个由办公室组成的集合。那么两个子公司A和B的最小距离定义为min(dist(x,y))(x∈A,y∈B)。其中dist(x,y)表示两个办公室之间的最短路径长度。
现在共有Q个询问,每次询问分别在两个子公司间的最小距离。
Input
对于每组数据:
第一行两个正整数N和M。城市编号为1至N,子公司编号为1至M。
接下来N-1行给定所有道路的两端城市编号和道路长度。
接下来M行,依次按编号顺序给出各子公司办公室所在位置,每行第一个整数G,表示办公室数,接下来G个数为办公室所在位置。
接下来一个整数Q,表示询问数。
接下来Q行,每行两个正整数a,b(a不等于b),表示询问的两个子公司。
【数据范围】
0<=边权<=100
1<=N,M,Q,工厂总数<=100000
Output
Sample Input
Sample Output
0
0
题意:
求树上最短路。
思路:
差不多裸的LCA,这里就当总结LCA的倍增实现了。
dfs求出每个节点的深度及到根节点的距离,并记录每个节点的第 (1<<i) 个祖先,这里和RMQ思想类似;
求LCA时先找到较深的节点(设为 u)与较浅的节点(设为 v)同高度的祖先,在由上至下找两节点第一个不同的祖先,则上一个祖先就是 u、v 的祖先。
代码:
#include<iostream>
#include<cstdio>
#include<vector>
#include<cstring>
using namespace std;
const int maxn = 1e5+5, INF = 2e9+7; class Edge
{
public:
int u, v, w; Edge(int a, int b, int c):u(a), v(b), w(c)
{
}
}; int n, m, q;
int deep[maxn], up[maxn][20], dis[maxn];
vector<Edge> edges;
vector<int> grap[maxn], factory[maxn]; void AddEdge(int u, int v, int w)
{
grap[u].push_back(edges.size());
edges.push_back(Edge(u, v, w));
grap[v].push_back(edges.size());
edges.push_back(Edge(v, u, w));
} void Init()
{
cin>>n>>m;
edges.clear();
memset(up, 0, sizeof(up));
for(int i=0; i<=n; ++i)
{
dis[i]=0;
factory[i].clear();
grap[i].clear();
deep[i]=0;
}
for(int i=1; i<n; ++i)
{
int u, v, w;
cin>>u>>v>>w;
AddEdge(u, v, w);
}
for(int i=1; i<=m; ++i)
{
int r, a;
cin>>r;
while(r--)
{
cin>>a;
factory[i].push_back(a);
}
}
} void Dfs(int u, int fa, int dep, int d)
{
deep[u]=dep;
dis[u]=d;
for(int i=1; i<20; ++i)//u 的第(1<<i)个祖先是 up[u][i-1] 的第(1<<(i-1))个祖先
up[u][i]=up[up[u][i-1]][i-1];
for(int i=0; i<grap[u].size(); ++i)
{
Edge& e=edges[grap[u][i]];
if(e.v==fa) continue;
up[e.v][0]=u;
Dfs(e.v, u, dep+1, d+e.w);
}
} int LCA(int u, int v)
{
if(deep[u] < deep[v]) swap(u, v);
int k=deep[u]-deep[v];
for(int i=0; i<20; ++i)//到同一高度
if((1<<i)&k)
u=up[u][i];
if(u!=v)
{
for(int i=19; i>=0; --i)
{
if(up[u][i]!=up[v][i])
{
u=up[u][i];
v=up[v][i];
}
}
u=up[u][0];
}
return u;
} int GetDist(int u, int v)
{
int lca=LCA(u, v);
return dis[u]+dis[v]-2*dis[lca];
} void Solve()
{
Dfs(1, -1, 0, 0);
cin>>q;
while(q--)
{
int a, b;
cin>>a>>b;
int ans=INF;
for(int i=0; i<factory[a].size(); ++i)
{
if(ans==0) break;
for(int j=0; j>factory[b].size(); ++j)
{
if(ans==0) break;
if(factory[a][i]==factory[b][j])
{
ans=0;
break;
}
}
}
for(int i=0; i<factory[a].size(); ++i)
{
if(ans==0) break;
for(int j=0; j<factory[b].size(); ++j)
{
if(ans==0) break;
int MinDist=GetDist(factory[a][i], factory[b][j]);
ans=min(ans, MinDist);
}
}
cout<<ans<<endl;
}
} int main()
{
int t;
cin>>t;
while(t--)
{
Init();
Solve();
}
return 0;
}
hdu6115 Factory (LCA + 倍增)的更多相关文章
- 【codevs2370】小机房的树 LCA 倍增
2370 小机房的树 时间限制: 1 s 空间限制: 256000 KB 题目等级 : 钻石 Diamond 题目描述 Description 小机房有棵焕狗种的树,树上有N个节点,节点标号为0 ...
- LCA倍增算法
LCA 算法是一个技巧性很强的算法. 十分感谢月老提供的模板. 这里我实现LCA是通过倍增,其实就是二进制优化. 任何一个数都可以有2的阶数实现 例如16可以由1 2 4 8组合得到 5可以由1 2 ...
- 洛谷 3379 最近公共祖先(LCA 倍增)
洛谷 3379 最近公共祖先(LCA 倍增) 题意分析 裸的板子题,但是注意这题n上限50w,我用的边表,所以要开到100w才能过,一开始re了两发,发现这个问题了. 代码总览 #include &l ...
- CodeVs.2370 小机房的树 ( LCA 倍增 最近公共祖先)
CodeVs.2370 小机房的树 ( LCA 倍增 最近公共祖先) 题意分析 小机房有棵焕狗种的树,树上有N个节点,节点标号为0到N-1,有两只虫子名叫飘狗和大吉狗,分居在两个不同的节点上.有一天, ...
- POJ.1986 Distance Queries ( LCA 倍增 )
POJ.1986 Distance Queries ( LCA 倍增 ) 题意分析 给出一个N个点,M条边的信息(u,v,w),表示树上u-v有一条边,边权为w,接下来有k个询问,每个询问为(a,b) ...
- POJ.1330 Nearest Common Ancestors (LCA 倍增)
POJ.1330 Nearest Common Ancestors (LCA 倍增) 题意分析 给出一棵树,树上有n个点(n-1)条边,n-1个父子的边的关系a-b.接下来给出xy,求出xy的lca节 ...
- LCA(倍增在线算法) codevs 2370 小机房的树
codevs 2370 小机房的树 时间限制: 1 s 空间限制: 256000 KB 题目等级 : 钻石 Diamond 题目描述 Description 小机房有棵焕狗种的树,树上有N个节点, ...
- LCA(最近公共祖先)——LCA倍增法
一.前人种树 博客:最近公共祖先 LCA 倍增法 博客:浅谈倍增法求LCA 二.沙场练兵 题目:POJ 1330 Nearest Common Ancestors 代码: const int MAXN ...
- POJ - 1330 Nearest Common Ancestors(dfs+ST在线算法|LCA倍增法)
1.输入树中的节点数N,输入树中的N-1条边.最后输入2个点,输出它们的最近公共祖先. 2.裸的最近公共祖先. 3. dfs+ST在线算法: /* LCA(POJ 1330) 在线算法 DFS+ST ...
- 次小生成树(LCA倍增)
算法: 求出MST之后枚举每条在MST之外的边 连上之后会出现环 找到环中除加上的边之外权值最大的边 删除该边之后得到一颗新树 做法: 利用LCA倍增地维护最小生成树上两点之间的最大边权 每次枚举在M ...
随机推荐
- docker注册&打包
在https://hub.docker.com上注册账号,新建reosity docker images命令列出dockerlist docker login命令登录:这其中出现了一个问题,登录不上去 ...
- @Autowried入门和源码分析
话不多说直接上代码: 声明一个接口userDao: package ioc.hello; public interface UserDao { public void test(); } 2个实现类: ...
- 2020年的100天——FLAGS
2020年的100天--FLAGS Reading <爱的博弈>-- 作者: 约翰·戈特曼 (John Gottman) / 娜恩·西尔弗 (Nan Silver) <人间失格> ...
- 刷题[De1CTF 2019]SSRF Me
前置知识 本题框架是flask框架,正好python面向对象和flask框架没怎么学,借着这个好好学一下 这里我直接听mooc上北京大学陈斌老师的内容,因为讲的比较清楚,直接把他的ppt拿过来,看看就 ...
- 刷题[HCTF 2018]WarmUp
解题思路 进入页面之后,一个大大的滑稽. 查看源码 查看源码发现有source.php .打开 发现还有一个hint.php.打开发现 由此可知是代码审计了 解题 代码审计 先看此段代码,大致意思是. ...
- jwt攻击手段
jwt 秘钥泄漏/敏感文件泄漏 在一些配置文件被我们可查的情况下,包括各种文件读取,源码泄漏,此时伪造身份就变得很容易了. 空加密算法 对于明文的加密算法通常为HS256,在jwt中分为三个部分,以点 ...
- 谷歌发布Flutter Alpha:支持Windows
老孟导读:Windows来了,Mac.Linux.Web还远吗? 本文翻译自https://medium.com/flutter/announcing-flutter-windows-alpha-33 ...
- 中部:执具 | R语言数据分析(北京邮电大学)自整理笔记
第5章工欲善其事.必先利其器 代码,是延伸我们思想最好的工具. 第6章基础编程--用别人的包和函数讲述自己的故事 6.1编程环境 1.R语言的三段论 大前提:计算机语言程序=算法+数据结构 小前提:R ...
- 在 Minecraft 中管理 Kubernetes 集群
原文链接:在 Minecraft 中管理 Kubernetes 集群 微软 2015 年收购 Minecraft 之后不久开源了一个项目叫 Dockercraft,这个项目当时看起来非常有趣,通过 D ...
- 代码上传多个git仓库,切换过remote后导致 can't update
问题描述: 因为代码上传到github和coding 切换了 git--> rmote的地址:后来update失败 问题解决: 重新配置git解决:按提示操作就好 git fetch git p ...