洛谷题解——P2814 家谱

该博客主要解析了一道洛谷上的题目——P2814家谱,涉及并查集数据结构的应用。题目要求根据输入的父子关系找到每个人的最早祖先。博主提供了详细的解题思路和AC代码,通过使用map<string, string>存储并查集,实现了字符串类型的节点管理和查询。在代码中,博主特别注意了初始化和数据读入的处理,确保了程序的正确运行。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

题目相关

题目链接

洛谷,https://www.luogu.com.cn/problem/P2814

MYOJ,http://47.110.135.197/problem.php?id=5344

题目描述

给出充足的父子关系,请你编写程序找到某个人的最早的祖先。

输入格式

输入由多行组成,首先是一系列有关父子关系的描述,其中每一组父子关系中父亲只有一行,儿子可能有若干行,用 #name 的形式描写一组父子关系中的父亲的名字,用 +name 的形式描写一组父子关系中的儿子的名字;接下来用 ?name 的形式表示要求该人的最早的祖先;最后用单独的一个 $ 表示文件结束。

输出格式

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

输入样例

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

输出样例

Edward Arthur
Walter Arthur
Rodney George
Arthur Arthur

数据范围

规定每个人的名字都有且只有 6 个字符,而且首字母大写,且没有任意两个人的名字相同。最多可能有 10^3 组父子关系,总人数最多可能达到 5×10^4 人,家谱中的记载不超过 30 代。

题解报告

题意分析

题意还是比较清晰的,不需要分析了。

看完后,就知道考察并查集的知识点。和其他并查集不同的地方,这里的数据类似都是字符串,而不是数字。

解题思路

还是套用并查集的模板。由于考虑到数据都是字符串,所以直接使用 map<string, string> 来保存并查集是最简单的。

样例数据分析

根据样例数据,我们可以写出这样的集合关系。

ds["Arthur"]="Arthur";
ds["Edward"]="Arthur";
ds["Gareth"]="Arthur";
ds["George"]="George";
ds["Rodney"]="George";
ds["Walter"]="Arthur";

下面我们根据 ds 来查询就可以得到答案。

技术难点

初始化 ds

由于使用 map<string, string> 来保存并查集,所以程序一开始的时候,没有办法进行初始化。我们需要在合适的地方对 ds 数组进行初始化。

我们可以在读入名字的时候,查询一下名字的父节点,如果父节点为空,说明没有初始化过,应该进行初始化。如果父节点不为空,说明已经初始化了。

数据读入

本题数据读入属于不知道长度类型,套用对应的模板代码即可。

AC 参考代码

//https://www.luogu.com.cn/problem/P2814
//P2814 家谱
#include <bits/stdc++.h>

using namespace std;

const int MAXN=5e4+4;
map<string, string> ds;

string find_root(string x) {
    return x==ds[x] ? x : ds[x]=find_root(ds[x]);
}

void union_set(string x, string y) {
    ds[y]=x;
}

int main() {
    char ch;
    string father;
    string s;

    while (cin>>ch) {
        if ('$'==ch) {
            break;
        }

        cin>>s;
        //初始化
        if (""==find_root(s)) {
            ds[s]=s;
        }

        if ('?'==ch) {
            //查询
            string t = find_root(s);
            cout<<s<<" "<<t<<"\n";
        } else if ('#'==ch) {
            father=s;
        } else if ('+'==ch) {
            //建立索引
            union_set(father, s);
        }
    }

    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

努力的老周

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值