家谱

Description

现代的人对于本家族血统越来越感兴趣, 现在给出充足的父子关系, 请你编写程序找到 某个人的最早的祖先。

Input

输入文件由多行组成, 首先是一系列有关父子关系的描述, 其中每一组父子关系由二行 组成,用#name 的形式描写一组父子关系中的父亲的名字,用+name 的形式描写一组父子关 系中的儿子的名字;接下来用?name 的形式表示要求该人的最早的祖先;最后用单独的一个 $表示文件结束。规定每个人的名字都有且只有 6 个字符,而且首字母大写,且没有任意两 个人的名字相同。最多可能有 1000 组父子关系,总人数最多可能达到 50000 人,家谱中的 记载不超过 30 代。

Output

按照输入文件的要求顺序,求出每一个要找祖先的人的祖先,格式:本人的名字+一个 空格+祖先的名字+回车。

Sample Input

#George
+Rodney
#Arthur
+Gareth
+Walter
#Gareth
+Edward
?Edward
?Walter
?Rodney
?Arthur
$

Sample Output

Edward Arthur
Walter Arthur
Rodney George
Arthur Arthur

题目分析

#:表示父亲
+:表示儿子(儿子上面离他最近的#就是他父亲)
?:查询他的爸爸是谁
$:结束程序

解题思路

一道很水的并查集,还有要用map
不懂并查集的请进(个人理解)

AC代码

#include<iostream>
#include<cstdio>
#include<map>
using namespace std;
map<string,string>fa;
string s,s1,ss;
string find(string s)//找爸爸
{
   
	if(fa[s]==s)return s;
	else return fa[s]=find(fa[s]);//路径压缩
}
int main()
{
   
	while(1)
	{
   
		cin>>s;
		s1="";//清空
		for(int i=1;i<s.size();i++)//取输入的字符串部分
		 s1+=s[i];
		if(s=="$")break;//结束程序
		if(s[0]=='#')//是父亲
		{
   
			ss=s1;//暂时存储
			if(fa[s1]=="")fa[s1]=s1;//如果他没父亲,那么他父亲就是自己
		}
		if(s[0]=='+')fa[s1]=ss;//是儿子就赋值父亲
		if(s[0]=='?')cout<<s1<<" "<<find(s1)<<endl;//找爸爸
	}
}

谢谢