#PAT#day7

入门篇(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
  1. 这里应该再设置一个新的数组,只存储每个年龄段的前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. 这道题就是用的位图法!
  2. ▲最后一个测试点一直通不过。因为并不是每个提交记录中得到满分之后就要将完全通过的次数+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
  1. 这段代码的第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
  1. 在school结构体中增加一个变量标记该学校录取的最后一个学生的排名,并将初值设为-1(0也可以,非自然数都可以),因为学生的排名都是自然数,故即使该学校一个录取学生都没有的时候,也不会出现凑巧排名相等而录取的情况!
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值