贤鱼的刷题日常(数据结构栈学习)-6377:生日相同 2.0--题目详解

🏆今日学习目标:
🍀例题讲解6377:生日相同 2.0
✅创作者:贤鱼
⏰预计时间:25分钟
🎉个人主页:贤鱼的个人主页
🔥专栏系列:c++
🍁贤鱼的个人社区,欢迎你的加入 贤鱼摆烂团

请添加图片描述

6377:生日相同 2.0

题目

描述
在一个有180人的大班级中,存在两个人生日相同的概率非常大,现给出每个学生的名字,出生月日。试找出所有生日相同的学生。

输入
第一行为整数n,表示有n个学生,n ≤ 180。此后每行包含一个字符串和两个整数,分别表示学生的名字(名字第一个字母大写,其余小写,不含空格,且长度小于20)和出生月(1 ≤ m ≤ 12)日(1 ≤ d ≤ 31)。名字、月、日之间用一个空格分隔
输出
每组生日相同的学生,输出一行,其中前两个数字表示月和日,后面跟着所有在当天出生的学生的名字,数字、名字之间都用一个空格分隔。对所有的输出,要求按日期从前到后的顺序输出。 对生日相同的名字,按名字从短到长按序输出,长度相同的按字典序输出。如没有生日相同的学生,输出”None”

  • 样例输入
6
Avril 3 2
Candy 4 5
Tim 3 2
Sufia 4 5
Lagrange 4 5
Bill 3 2
  • 样例输出
3 2 Tim Bill Avril
4 5 Candy Sufia Lagrange

思路

题目要求按照生日相同,我们用结构体储存所有数据,sort快速排序,排序内容首先按照月份从小到大排序,接着按照同样月份排序生日,最后处理一下名字部分就好了

大概思路就是这样,接下来仔细分析分析

  • 题目要求生日相同
    排序月份和日以后可以让所有的生日相同的人排成一队
    相当于从开头到这个子段的结尾都是同一个生日的人
    我们用栈来储存这个内容
    如果遇到下一个人和上一个人月份不同并且栈内不止一个人
    输出,并且清空栈

如何判断只有一个人(没有相同生日)?
按照循环,如果当前月份和前面一个人月份不同,出栈,如果栈为空,就入栈当前循环的这个人所有信息,然后continue,如果continue前不入栈,会拉下个大怨种

  • 如何输出
    首先排序姓名长度,接着使用compare函数比较字典序,完事输出

AC代码

#include<cmath>
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<stack>
using namespace std;
struct node{
	string name;
	int y;
	int d;
	bool operator <(const node &b)const{
		 return y<b.y;
	}
}a[500];
int p=0;
stack<string>name;
stack<int>y;
stack<int>d;
int ecnt=0;
string nm[1005];
int yu[1005];
int da[1005];
void pd(){
	memset(yu,0,sizeof(yu));
	memset(da,0,sizeof(da));
	while(!name.empty()){
		nm[++ecnt]=name.top();
		name.pop();
	}
	if(ecnt==1){
		ecnt=0;
		y.pop();
		d.pop();
		return ;
	}
	for(int i=1;i<=ecnt;i++){
		for(int j=i+1;j<=ecnt;j++){
			if(nm[i].length()>nm[j].length()){
				swap(nm[i],nm[j]);
			}
		}
	}
	for(int i=1;i<=ecnt;i++){
		for(int j=i+1;j<=ecnt;j++){
			if(nm[i].length()>nm[j].length()){
				swap(nm[i],nm[j]);
			}else if(nm[i].compare(nm[j])>0&&nm[i].length()==nm[j].length()){
				swap(nm[i],nm[j]);
			}
		}
	}
	cout<<y.top()<<" "<<d.top()<<" ";
	for(int i=1;i<=ecnt;i++){
		cout<<nm[i]<<" ";
	}
	cout<<endl;
	while(!y.empty()){
		y.pop();
		d.pop();
	}
	ecnt=0;
	p=1;
}
int main(){
	int n;
	cin>>n;
	for(int i=1;i<=n;i++){
		cin>>a[i].name>>a[i].y>>a[i].d;
	}
	sort(a+1,a+1+n);
	for(int i=1;i<=n;i++){
		for(int j=i+1;j<=n;j++){
			if(a[i].y==a[j].y&&a[i].d>a[j].d){
				swap(a[i],a[j]);
			}
		}
	}
	name.push(a[1].name);
	y.push(a[1].y);
	d.push(a[1].d);
	for(int i=2;i<=n;i++){	
		if(name.empty()){
			name.push(a[i].name);
			y.push(a[i].y);
			d.push(a[i].d);
		}else{
			if(a[i].y!=y.top()||a[i].d!=d.top()){
				string nn;
				if(!name.empty()){
					nn=name.top();
					name.pop();
				}
				if(name.empty()){
					name.push(a[i].name);
					y.push(a[i].y);
					d.push(a[i].d);
					continue;
				}
				name.push(nn);
				pd();
				name.push(a[i].name);
				y.push(a[i].y);
				d.push(a[i].d);
			}
			else{
				name.push(a[i].name);
				y.push(a[i].y);
				d.push(a[i].d);
			}
		}
	}
	string mn;
	if(!name.empty()){
		mn=name.top();
		name.pop();
	}
	if(name.empty()){
		if(!p) cout<<"None"<<endl;
		return 0;
	}
	name.push(mn);
	pd();
	return 0;
}

请添加图片描述

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

贤鱼不闲

一分钱也是爱!!!

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

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

打赏作者

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

抵扣说明:

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

余额充值