HDU 3786 找出直系亲属 (并查集/DFS)

找出直系亲属

Description

如果A,B是C的父母亲,则A,B是C的parent,C是A,B的child,如果A,B是C的(外)祖父,祖母,则A,B是C的grandparent,C是A,B的grandchild,如果A,B是C的(外)曾祖父,曾祖母,则A,B是C的great-grandparent,C是A,B的great-grandchild,之后再多一辈,则在关系上加一个great-。

Input

输入包含多组测试用例,每组用例首先包含2个整数n(0<=n<=26)和m(0<m<50), 分别表示有n个亲属关系和m个问题, 然后接下来是n行的形式如ABC的字符串,表示A的父母亲分别是B和C,如果A的父母亲信息不全,则用-代替,例如A-C,再然后是m行形式如FA的字符串,表示询问F和A的关系。 
当n和m为0时结束输入。

Output

如果询问的2个人是直系亲属,请按题目描述输出2者的关系,如果没有直系关系,请输出-。 
具体含义和输出格式参见样例.

Sample Input

3 2

ABC

CDE

EFG

FA

BE

0 0

Sample Output

great-grandparent

-

两种解题方法:

1.深搜(DFS):dp[i][2]数组向上遍历其父母节点

#include<iostream>
#include<sstream>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#include<cstring>
#include<cstdio>
#include<map>
#include<vector>
#include<stack>
#include<queue>
#include<set>
#include<list>
#define mod 998244353
#define Max 0x3f3f3f3f
#define Min 0xc0c0c0c0
#define mst(a) memset(a,0,sizeof(a))
#define f(i,a,b) for(int i=a;i<b;i++)
using namespace std;
typedef long long ll;
const int maxn='Z'+1;
char dp[maxn][2];   //dp记录每个节点的父节点和母节点dp[i][0],dp[i][1]
void init(){
    for(int i='A';i<maxn;i++){
        dp[i][0]='-';   //初始化每个节点的父母节点为-
        dp[i][1]='-';
    }
}
int dfs(char sx,char ex,int step){
    if(sx=='-'){        //某次遍历结束,未找到ex,,则返回int最大值(无父节点:dfs1递归,无母节点:dfs2递归)
        return INT_MAX;     //int的最大值
    }
    if(sx==ex){         //遍历找到ex则返回step关系层数
        return step;
    }
    int a,b;
    a=dfs(dp[sx][0],ex,step+1);     //dfs1,从sx的父节点开始向根遍历,
    b=dfs(dp[sx][1],ex,step+1);     //dfs2,从sx的母节点开始向根遍历
    return min(a,b);                //取两次dfs的最小遍历次数,即为关系层数
}
void show(int step,int mark){       //输出结果
    if(mark==1){
        if(step==1){
            cout<<"child"<<endl;
        }
        else{
            while(step>2){
                cout<<"great-";
                step--;
            }
            cout<<"grandchild"<<endl;
        }
    }
    else{
         if(step==1){
            cout<<"parent"<<endl;
        }
        else{
            while(step>2){
                cout<<"great-";
                step--;
            }
            cout<<"grandparent"<<endl;
        }
    }
}
int main(){
    ios::sync_with_stdio(false);
    int n,m,val,flag;
    while(cin>>n>>m){
        if(n==0 && m==0){
            break;
        }
        init();
        char s1,s2,s3;
        for(int i=0;i<n;i++){
            cin>>s1>>s2>>s3;
            dp[s1][0]=s2;
            dp[s1][1]=s3;
        }
        for(int j=0;j<m;j++){
            cin>>s1>>s2;
            flag=1;
            val=dfs(s1,s2,0);               //判断s2是否为s1的父母节点
            if(val == INT_MAX){                  //s2不是s1的父母节点
                flag=0;
                val=dfs(s2,s1,0);        //则反过来判断s1是否为s2的父母节点
            }
            if(val == INT_MAX){                 //s2与s1无直系亲属关系
                cout<<"-"<<endl;
                continue;
            }
            show(val,flag);         //s2与s1存在直系亲属关系,打印输出
        }
    }
    return 0;
}

2.并查集:child[i]向下遍历其孩子节点

#include<iostream>
#include<sstream>
#include<cstdlib>
#include<cmath>
#include<algorithm>
#include<cstring>
#include<cstdio>
#include<map>
#include<vector>
#include<stack>
#include<queue>
#include<set>
#include<list>
#define mod 998244353
#define Max 0x3f3f3f3f
#define Min 0xc0c0c0c0
#define mst(a) memset(a,0,sizeof(a))
#define f(i,a,b) for(int i=a;i<b;i++)
using namespace std;
typedef long long ll;
using namespace std;
int child[100];      //孩子节点数组
void init(){        //初始化每个节点的孩子节点
    for(int i=0;i<100;i++){
        child[i]=i;
    }
}
int Find(int x,int y){      //关键函数,查找y是否x的孩子节点
	while(1){
		if(x==y){
            return 1;
		}
		if(x==child[x]){
          return 0;
		}
		x=child[x];
	}
}
int main(){
	int n,m;
	while(cin>>n>>m){
        if(n==0&&m==0){
            break;
        }
        init();
        while(n--){
            char s1,s2,s3;
            cin>>s1>>s2>>s3;
            if(s2!='-'){
                child[s2-'A']=s1-'A';     //保存孩子节点
            }
            if(s3!='-') {
                child[s3-'A']=s1-'A';
            }
        }
        while(m--){
            char s1,s2;
            cin>>s1>>s2;
            int val1=s1-'A',val2=s2-'A';
            if(Find(val1,val2)){      //s2为s1的孩子节点
                int step=0;
                while(val1!=val2){      //遍历
                    val1=child[val1];
                    step++;            //遍历次数累加为关系层数
                }
                if(step==1){
                    cout<<"parent"<<endl;
                }
                else{
                    while(step>2){
                        cout<<"great-";
                        step--;
                    }
                    cout<<"grandparent"<<endl;
                }
            }
            else if(Find(val2,val1)){       //s1为s2的孩子节点
                int step=0;
                while(val2!=val1){          //遍历
                    val2=child[val2];
                    step++;                 //遍历次数累加为关系层数
                }
                if(step==1){
                    cout<<"child"<<endl;
                }
                else{
                    while(step>2){
                        cout<<"great-";
                        step--;
                    }
                    cout<<"grandchild"<<endl;
                }
            }
            else {                  //s1,s2无直系亲属关系
                cout<<'-'<<endl;
            }
        }
    }
	return 0;
}

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
智慧校园整体解决方案是响应国家教育信息化政策,结合教育改革和技术创新的产物。该方案以物联网、大数据、人工智能和移动互联技术为基础,旨在打造一个安全、高效、互动且环保的教育环境。方案强调从数字化校园向智慧校园的转变,通过自动数据采集、智能分析和按需服务,实现校园业务的智能化管理。 方案的总体设计原则包括应用至上、分层设计和互联互通,确保系统能够满足不同用户角色的需求,并实现数据和资源的整合与共享。框架设计涵盖了校园安全、管理、教学、环境等多个方面,构建了一个全面的校园应用生态系统。这包括智慧安全系统、校园身份识别、智能排课及选课系统、智慧学习系统、精品录播教室方案等,以支持个性化学习和教学评估。 建设内容突出了智慧安全和智慧管理的重要性。智慧安全管理通过分布式录播系统和紧急预案一键启动功能,增强校园安全预警和事件响应能力。智慧管理系统则利用物联网技术,实现人员和设备的智能管理,提高校园运营效率。 智慧教学部分,方案提供了智慧学习系统和精品录播教室方案,支持专业级学习硬件和智能化网络管理,促进个性化学习和教学资源的高效利用。同时,教学质量评估中心和资源应用平台的建设,旨在提升教学评估的科学性和教育资源的共享性。 智慧环境建设则侧重于基于物联网的设备管理,通过智慧教室管理系统实现教室环境的智能控制和能效管理,打造绿色、节能的校园环境。电子班牌和校园信息发布系统的建设,将作为智慧校园的核心和入口,提供教务、一卡通、图书馆等系统的集成信息。 总体而言,智慧校园整体解决方案通过集成先进技术,不仅提升了校园的信息化水平,而且优化了教学和管理流程,为学生、教师和家长提供了更加便捷、个性化的教育体验。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值