这个题目表面上是一个字符串排序问题  但处理起来有些复杂

 

因为题目只是告诉了 两两字符串之间的对应关系 

 

所以本题的处理策略是  找出那个第一任女友 第一任女友有个特点就是她只出现在左边

 

找出第一任女友的前提是把所有的字符串输入到mm数组中 并用link数组记录输入的每一行的两个女孩在mm数组中的位置

 

link[i][0] link[i][1]分别是记录在左边和右边的女孩  用in[link[i][1]]++ 然后找出in[i]==0的那个i 就是第一任女友 

因为她从未在右边出现过   之后利用fun(i)  fun函数的作用是按女孩出现的先后顺序给这些女孩赋值  mm[i]是第一个

出现的  所以遍历link[i][0] 找出第一任女友出现的每一行  给每一行的右边的女孩y 赋上 max(length[x] ,length[x]+1)

的值 之后fun(y) 找出每一个比y大的人 进行赋值  最终结束循环后  所有的字符串都被赋上了 正确的值

 

 

 

就像是一个横向的二叉树一样 (可能比喻不是很贴切)

但这是一个很棒的排序思路

实现起来可能比较复杂  所以需要细心 一些 耐心一些 

本人水平不足  欢迎大家批评指正

f2deb48f8c5494eeb37973f625f5e0fe98257e81.jpg
参考代码:

#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std; 
char mm[105][15];  
char s1[15],s2[15];  
int link[105][2];  
int length[105]={0};  
int in[105];  
int N;  
struct girl 
{  
    char name[15];  
    int num;  
}g[105];  //记录最后的结果 包括女孩的名字 及女孩成为女朋友的先后顺序 
void fun(int x) //给每个字符串按照先后顺序赋上相应的值 
{  
    int i,y;  
    for(i=0;i<N;i++)  
        if(link[i][0]==x)  //在左边找到了该字符串 
        {  
            y=link[i][1];  //y是它的右边字符串 
            length[ y ]=length[x]+1>=length[y]?length[x]+1:length[y];  //右边字符串的值为max(左边字符串的值+1,本身字符串的值) 
            fun(y);  //接着寻找当它作为左边字符串时的情况  
        }  
}  
int comp(girl a, girl b)  
{  
    return a.num<b.num ;//按从小到大排序 
}  
int main()  
{  
    int i,j,T,n,f,x,y;   
    cin>>T;//测试数据的组数 
    while(T--)  
    {  
        memset(mm,0,sizeof(mm));//该数组保存输入的女孩的名字  同时初始化所有的数组  
        memset(link,0,sizeof(link)); //该数组记录每行输入的两个女孩的名字在mm数组中的位置 
        memset(in,0,sizeof(in));  //该数组记录女孩的名字在右边出现过的次数 
        memset(length,0,sizeof(length));  //按女孩出现的先后顺序给女孩赋值 
        cin>>N; //每组测试数据下的需要比较的行数 即两个MM之间进行比较 
        n=0;  //mm数组中元素的个数 也即是mm的个数 
        for(i=0;i<N;i++)  
        {  
            scanf("%s%s",s1,s2);  //输入两个MM的名字 
            f=0;  
            for(j=0;j<n;j++)  
                if(strcmp(s1,mm[j])==0)  //如果mm数组里面已经有了输入的第一个MM的名字  那么用 x记录下这个MM名字的位置 
                {  
                    f=1;  
                    break;  
                }  
            if(f==1)  
                x=j;  //x记录mm名字的位置 
            else  //如果mm数组里面没有即将输入的第一个MM的名字  则将该字符串输入到mm数组中  同时记录该字符串的位置 
            {  
                strcpy(mm[n++],s1);  
                x=n-1;  //记录字符串的位置 
            }  
            f=0;  
            for(j=0;j<n;j++)  //同理记录第二个MM的名字 如果mm数组中已经有了该MM的名字  记录该字符串的位置 
                if(strcmp(s2,mm[j])==0)  
                {  
                    f=1;  
                    break;  
                }  
            if(f==1)  
                y=j;  //记录第二个mm名字的位置 
            else  
            {  
                strcpy(mm[n++],s2);  //否则将该MM的名字输入到mm数组中去 
                y=n-1;  
            }  
            link[i][0]=x;  //记录下每一行输入的两个MM的名字在数组中的位置,i代表行数 
            link[i][1]=y;  //每一行的字符串S1,s2在mm数组中的位置 
        }  
        for(i=0;i<N;i++)  
            in[link[i][1]]++;  //以在右边的小姐姐在mm数组中的位置 作为in数组中的下标   来保存该MM在右边出现的次数 
        for(i=0;i<n;i++)  
            if(in[i]==0)  //找到第一个没有出现在右边的字符串(即为第一任女友) 记录下它的下标 
                break;  
        fun(i);  //将该下标作为函数的实参 
        for(i=0;i<n;i++)  
        {  
            strcpy(g[i].name,mm[i]);  //将mm数组以及length数组复制到g[i]数组中 
            g[i].num=length[i];  
        }         
        sort(g,g+n,comp);  //对数组元素以num的值进行从小到大的排序 
        for(i=0;i<n;i++)  
        {  
            printf("%s",g[i].name);  //输出女朋友出现的先后次序 
            if(i==n-1)  
                printf("\n");  
            else  
                printf(" ");  
        }      
    }  
    return 0;  
}


点赞(8)
 

0.0分

5 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 1 条评论

Plaire7 5年前 回复TA
太厉害了你