文章目录
入门篇(2)——算法初步
1055 The World’s Richest
我的代码
#include<stdio.h>
#include<algorithm>
#include<string.h>
using namespace std;
struct Peo{
char name[9];
int age;
int worth;
}peo[100100];
bool cmp(Peo a,Peo b){
if(a.worth!=b.worth) return a.worth>b.worth;
else if(a.age!=b.age) return a.age<b.age;
else return strcmp(a.name,b.name)<0;
}
int main(){
int n,k;
scanf("%d%d",&n,&k);
for(int i=0;i<n;i++){
scanf("%s %d %d",peo[i].name,&peo[i].age,&peo[i].worth);
}
sort(peo,peo+n,cmp);
for(int i=0;i<k;i++){
int flag=0,m,max,min;
scanf("%d%d%d",&m,&min,&max);
printf("Case #%d:\n",i+1);
for(int j=0,r=0;j<n&&r<m;j++){
if(peo[j].age>=min&&peo[j].age<=max){
flag=1;
printf("%s %d %d\n",peo[j].name,peo[j].age,peo[j].worth);
r++;
}
}
if(flag==0) printf("None\n");
}
}
notes
- 这里应该再设置一个新的数组,只存储每个年龄段的前100个人。因为题目写了M<100,即100个之后的人永远不会被输出。这样可以降低查询的复杂度。(存入前100个人的过程可以设置一个age[]数组标识每个年龄的人数,即位图法)
1075 PAT Judge
我的代码
#include<stdio.h>
#include<algorithm>
#include<string.h>
using namespace std;
struct User{
long user_id;
int highest_score[6]={-2,-2,-2,-2,-2};
int total=0;
int prefect=0;
int flag=-1;//还不能打印
int rank=1;
}user[10100],user2[10100];
int score[6];
bool cmp(User a,User b){
if(a.total!=b.total) return a.total>b.total;
else if(a.prefect!=b.prefect) return a.prefect>b.prefect;
else return a.user_id<b.user_id;
}
int main(){
int n,k,max=0;//n个用户,k个问题,m条记录,max编号最大的用户
long m;
scanf("%d%d%ld",&n,&k,&m);
for(int i=0;i<k;i++){
scanf("%d",&score[i]);
}
for(long i=0;i<m;i++){
long temp_id;
int pro_id,part_score;
scanf("%ld%d%d",&temp_id,&pro_id,&part_score);
user[temp_id-1].user_id=temp_id;
if(max<temp_id) max=temp_id;
if(part_score>=0) user[temp_id-1].flag=1;
if(part_score==score[pro_id-1]&&user[temp_id-1].highest_score[pro_id-1]<score[pro_id-1]) user[temp_id-1].prefect++;
if(user[temp_id-1].highest_score[pro_id-1]<part_score){
user[temp_id-1].highest_score[pro_id-1]=part_score;
}
}
for(int i=0;i<max;i++){
for(int j=0;j<k;j++){
if(user[i].highest_score[j]>0)
user[i].total+=user[i].highest_score[j];
}
}
sort(user,user+max,cmp);
user[0].rank=1;
for(int i=1;i<n;i++){
if(user[i].total!=user[i-1].total) user[i].rank=i+1;
else user[i].rank=user[i-1].rank;
}
for(int i=0;i<n;i++){
if(user[i].flag==1){
printf("%d %05ld %d",user[i].rank,user[i].user_id,user[i].total);
for(int j=0;j<k;j++){
if(user[i].highest_score[j]>=0){
printf(" %d",user[i].highest_score[j]);
}
else if(user[i].highest_score[j]==-1) printf(" 0");
else printf(" -");
}
printf("\n");
}
}
}
notes
- 这道题就是用的位图法!
- ▲最后一个测试点一直通不过。因为并不是每个提交记录中得到满分之后就要将完全通过的次数+1,只有第一次才+1。比如说我本人也经常反复提交完全通过的答案。
1080 Graduate Admission
我的代码(26/30部分正确版本)
#include<stdio.h>
#include<string.h>
#include<algorithm>
using namespace std;
struct App{
int ge,gi,g_final,rank,id;
int prefer_school[5];
}app[40010];
struct School{
int quota;
int stu[40010];
int num_stu=0;
}school[100];
bool cmp(App a,App b){
if(a.g_final!=b.g_final) return a.g_final>b.g_final;
else return a.ge>b.ge;
}
int main(){
int n,m,k;
scanf("%d%d%d",&n,&m,&k);
//N (¡<=40,000), the total number of applicants;
//M (¡<=100), the total number of graduate schools;
//K (¡<=5), the number of choices an applicant may have.
for(int i=0;i<m;i++){
scanf("%d",&school[i].quota);
}
//for(int i=0;i<m;i++) printf("%d \n",school[i].quota);
for(int i=0;i<n;i++){
scanf("%d%d",&app[i].ge,&app[i].gi);
app[i].g_final=app[i].ge+app[i].gi;
app[i].id=i;
for(int j=0;j<k;j++){
scanf("%d",&app[i].prefer_school[j]);
}
}
sort(app,app+n,cmp);
app[0].rank=1;
for(int i=1;i<n;i++){
if(app[i].g_final<app[i-1].g_final){
app[i].rank=i+1;
}
else if(app[i].ge<app[i-1].ge){
app[i].rank=i+1;
}
else{
app[i].rank=app[i-1].rank;
}
}
app[n].rank=app[n-1].rank+1;
//for(int i=0;i<n;i++) printf("%d %d\n",app[i].rank,app[i].id);
for(int i=0;i<n;i++){
for(int j=0;j<k;j++){
if(school[app[i].prefer_school[j]].num_stu<school[app[i].prefer_school[j]].quota){
school[app[i].prefer_school[j]].stu[school[app[i].prefer_school[j]].num_stu]=app[i].id;
school[app[i].prefer_school[j]].num_stu++;
break;
}
else if(app[i].rank==app[school[app[i].prefer_school[j]].stu[school[app[i].prefer_school[j]].num_stu-1]].rank){
school[app[i].prefer_school[j]].stu[school[app[i].prefer_school[j]].num_stu]=app[i].id;
school[app[i].prefer_school[j]].num_stu++;
break;
}
}
//printf("%d\n",school[app[i].prefer_school[j]].num_stu);
}
//printf("%d",school[0].stu[1]);
for(int i=0;i<m;i++){
sort(school[i].stu,school[i].stu+school[i].num_stu);
for(int j=0;j<school[i].num_stu;j++){
printf("%d",school[i].stu[j]);
if(j<school[i].num_stu-1) printf(" ");
}
printf("\n");
}
}
notes
- 这段代码的第64行
else if(app[i].rank==app[school[app[i].prefer_school[j]].stu[school[app[i].prefer_school[j]].num_stu-1]].rank)
有错误。虽然我的思路是对的——判断该生的排名与心仪学校所录取的最后一个学生的排名是否相同,但是这种-1的写法没有考虑有的学校录取名额为0的情况。即,该学校一个同学都没有录取,那么-1之后当前学生的比较对象变为stu[-1],这是不存在的。
更改之后的代码
notes
- 在school结构体中增加一个变量标记该学校录取的最后一个学生的排名,并将初值设为-1(0也可以,非自然数都可以),因为学生的排名都是自然数,故即使该学校一个录取学生都没有的时候,也不会出现凑巧排名相等而录取的情况!