洛谷题解——P2814 家谱

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

题目相关

题目链接

洛谷,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;
}
洛谷P1554题为“梦中的统计”。有多位用户分享了该题的答案及解析,旨在帮助初学者。部分答案以C语言编写,也有C++和Python3的题解。 C语言题解思路是对比1 - 9个数字分别出现的次数,使用两个循环解决,关键在于对比每个数的位数上的数。以下是C语言代码示例: ```c #include<stdio.h> int main() { int m,n,ans[]={0,0,0,0,0,0,0,0,0,0}; scanf("%d%d",&m,&n); for(int i=m;i<=n;i++){ int j=i; while(j!=0) { int mod=j%10; // 取余对比每个数 ans[mod]++; j/=10; } } for(int i=0;i<10;i++){ printf("%d ",ans[i]); } } ``` C++题解采用暴力枚举n - m的每个数再统计的方法,代码如下: ```cpp #include<bits/stdc++.h> using namespace std; int n,m,a[15]; int main(){ scanf("%d%d",&n,&m); for(int i=n;i<=m;i++){ int tmp=i,k=0,fs; while(tmp!=0){ ++k; fs=tmp%10; tmp/=10; a[fs]++; } } for(int i=0;i<=9;i++) printf("%d ",a[i]); puts(""); return 0; } ``` Python3题解通过取余统计每个数字出现的次数,代码如下: ```python m,n = map(int,input().split()) list1 = [0*i for i in range(10)] for i in range(m,n+1): while i != 0: num = i % 10 list1[num] += 1 i = i//10 for i in list1: print(i,end=' ') ``` 此外,还有从蓝桥云课、洛谷、AcWing等知名刷题平台精心挑选题目并结合算法标签和难度等级进行系统分类的资料,涵盖从基础到进阶的多种算法和数据结构,洛谷P1554题是其中的必刷题目之一,旨在为不同阶段的编程学习者提供学习提升路径[^1][^2][^3][^4][^5]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

努力的老周

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

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

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

打赏作者

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

抵扣说明:

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

余额充值