欢迎使用CSDN-markdown编辑器
2015-07-15 09:44
267 查看
hiho1067最近公共祖先·二
tarjan离线算法的教学题,不多说什么,只是这题是字符串,蛋疼的要死啊(还好有map)。。。。。。lca离线实质上就是dfs+并查集#1067 : 最近公共祖先·二
时间限制:10000ms
单点时限:1000ms
内存限制:256MB
描述
上上回说到,小Hi和小Ho用非常拙劣——或者说粗糙的手段山寨出了一个神奇的网站,这个网站可以计算出某两个人的所有共同祖先中辈分最低的一个是谁。远在美国的他们利用了一些奇妙的技术获得了国内许多人的相关信息,并且搭建了一个小小的网站来应付来自四面八方的请求。
但正如我们所能想象到的……这样一个简单的算法并不能支撑住非常大的访问量,所以摆在小Hi和小Ho面前的无非两种选择:
其一是购买更为昂贵的服务器,通过提高计算机性能的方式来满足需求——但小Hi和小Ho并没有那么多的钱;其二则是改进他们的算法,通过提高计算机性能的利用率来满足需求——这个主意似乎听起来更加靠谱。
于是为了他们第一个在线产品的顺利运作,小Hi决定对小Ho进行紧急训练——好好的修改一番他们的算法。
而为了更好的向小Ho讲述这个问题,小Hi将这个问题抽象成了这个样子:假设现小Ho现在知道了N对父子关系——父亲和儿子的名字,并且这N对父子关系中涉及的所有人都拥有一个共同的祖先(这个祖先出现在这N对父子关系中),他需要对于小Hi的若干次提问——每次提问为两个人的名字(这两个人的名字在之前的父子关系中出现过),告诉小Hi这两个人的所有共同祖先中辈分最低的一个是谁?
提示一:老老实实分情况讨论就不会出错的啦!
提示二:并查集其实长得很像一棵树你们不觉得么?
输入
每个测试点(输入文件)有且仅有一组测试数据。
每组测试数据的第1行为一个整数N,意义如前文所述。
每组测试数据的第2~N+1行,每行分别描述一对父子关系,其中第i+1行为两个由大小写字母组成的字符串Father_i, Son_i,分别表示父亲的名字和儿子的名字。
每组测试数据的第N+2行为一个整数M,表示小Hi总共询问的次数。
每组测试数据的第N+3~N+M+2行,每行分别描述一个询问,其中第N+i+2行为两个由大小写字母组成的字符串Name1_i, Name2_i,分别表示小Hi询问中的两个名字。
对于100%的数据,满足N<=10^5,M<=10^5, 且数据中所有涉及的人物中不存在两个名字相同的人(即姓名唯一的确定了一个人),所有询问中出现过的名字均在之前所描述的N对父子关系中出现过,第一个出现的名字所确定的人是其他所有人的公共祖先。
输出
对于每组测试数据,对于每个小Hi的询问,按照在输入中出现的顺序,各输出一行,表示查询的结果:他们的所有共同祖先中辈分最低的一个人的名字。
#include<iostream> #include<cstdio> #include<algorithm> #include<cstring> #include<vector> #include<map> using namespace std; int n,m,cime,g[100005],cnt,f[100005],cl[100005]; struct edge{ int next,to; }; edge e[200005]; map<string,int> f1; map<int,string> f2; int q[100005]; vector<int> xw[100005]; void addedge(int from,int to) { cnt++;e[cnt].to=to;e[cnt].next=g[from];g[from]=cnt; cnt++;e[cnt].to=from;e[cnt].next=g[to];g[to]=cnt; return ; } int find(int x) { if (x!=f[x]) f[x]=find(f[x]); return f[x]; } void init() { cin>>n; for (int i=1;i<=n;i++) f[i]=i; string fa,son; for (int i=1;i<=n;i++) { cin>>fa; if (!f1[fa]) { cime++; f1[fa]=cime; f2[cime]=fa; } cin>>son; if (!f1[son]) { cime++; f1[son]=cime; f2[cime]=son; } addedge(f1[fa],f1[son]); } cin>>m; for (int i=1;i<=m;i++) { cin>>fa>>son; xw[f1[fa]].push_back(f1[son]);xw[f1[fa]].push_back(i); xw[f1[son]].push_back(f1[fa]);xw[f1[son]].push_back(i); } return ; } void update(int x) { for (int i=0;i<xw[x].size();i+=2) { if (cl[xw[x][i]]==0) continue; if (cl[xw[x][i]]==1) q[xw[x][i+1]]=xw[x][i]; if (cl[xw[x][i]]==2) q[xw[x][i+1]]=find(xw[x][i]); } return ; } void lca(int x,int fa) { for (int i=g[x];i;i=e[i].next) { if (e[i].to==fa) continue; cl[e[i].to]=1; lca(e[i].to,x); update(e[i].to); f[e[i].to]=x; cl[e[i].to]=2; } return ; } int main() { cime=cnt=0; memset(f,0,sizeof(f)); memset(g,0,sizeof(g)); memset(cl,0,sizeof(cl)); init(); cl[1]=1; for (int i=g[1];i;i=e[i].next) { cl[e[i].to]=1; lca(e[i].to,1); update(e[i].to); f[e[i].to]=1; cl[e[i].to]=2; } for (int i=1;i<=m;i++) cout<<f2[q[i]]<<endl; return 0; }
相关文章推荐
- SharePoint 2013 PowerShell命令备份还原报错
- Android Studio 使用笔记:[转] Mac下修改Android Studio 所用的JDK版本
- 记录那些奋斗的日子
- 学习重点:1、金典的设计模式在实际中应用2、JVM原理3、jui源代码
- performSegueWithIdentifier:sender里边的sender是啥意思
- 揭秘微商2015年的高端玩法
- TCP/IP-----------tcp粘包问题
- 函数指针作为函数参数,实现冒泡排序的升序排序和降序排序
- 动画特效一:图片翻转
- JSBinding + SharpKit / 常见问题
- 第一次到这个网站
- 解决NDK开发中Eclipse报错“Unresolved inclusion jni.h”的最终方法
- [架构]--系统架构演化之路
- ubuntu 操作命令
- Eclipse下导入外部jar包的3种方式
- 实战EmptyView,为ListView的加载缓冲
- 备份电脑系统的操作
- Eclipse项目名出现红叉
- centos7.1下载
- [刷题]Singleton