蓝桥杯省赛备战笔记—— (三)使用sort排序

需要载入头文件

#include<algorithm>

一、使用

#include<iostream>
#include<algorithm>
using namespace std;
int main(){
    int arr[] = {2,4,5,3,1};
    //sort(arr,arr+5);
    sort(arr + 1,arr + 4);
    for(int i = 0;i < 5;i++)
        cout <<arr[i]<<' ';
    return 0;
}
  1. sort(arr,arr+5)   ———— (数组名,数组名+长度) —— 将整个数组从小到大排序
  2. sort(arr + i,arr + j) ———— 将arr[ i ] 到 arr[ j - 1 ] 排序,其余元素保持原位置不变

如果希望arr 中的元素 从大到小排序(或按照某一个规则排列),传入第3个参数 —— ”排序方法“

sort(arr,arr  + 5, greater<int>());
  • <int> 表示待排序的数组中的元素类型为 int 

 

例题:前K名的平均成绩

#include<iostream> // 要载入这个头文件,否则dec_c++上会报错'greater'was not declared in this scope
#include<cstdio>
#include<algorithm>
using namespace std;
int s[35];
int main(){
    int n;
    scanf("%d",&n);
    for(int i = 0; i < n;i++){
        scanf("%d",&s[i]);
    }
    sort(s,s+n,greater<int>());
    int k;
    scanf("%d",&k);
    int sum = 0;
    for(int i = 0; i < k;i++){
        sum += s[i];
    }
    printf("%.2f", 1.0 * sum / k);    
    return 0;
}

 

例题:分数段统计

 

 

 

#include<stdio.h>
#include<iostream>
#include<algorithm>
using namespace std;
int a[35];
int b[7];
int main(){
    int n;
    scanf("%d",&n);
    for(int i = 0; i < n;i++){
        scanf("%d",&a[i]);
    }
    sort(a,a+n,greater<int>());
    for(int i = 0; i < n;i++){
        printf("%d\n",a[i]);
    }
    for(int i = 0; i < n;i++){
        if(a[i] == 100){
            b[1]++;
        }else if(a[i] >= 90 && a[i] <= 99 ){
            b[2]++;
        }else if(a[i] >= 80 && a[i] <= 89){
            b[3]++;
        }else if(a[i] >= 70 && a[i] <= 79){
            b[4]++;
        }else if(a[i] >= 60 && a[i] <= 69){
            b[5]++;
        }else{
            b[6]++;
        }
    }    
    for(int i = 1;i < 7;i++){
        if(i != 6)
            printf("%d ",b[i]);
        else
            printf("%d",b[i]);
    }    
    return  0;
}

 

二、 第三个参数—— 自定义排序方法

 在sort 第三个参数中传入 cmp ,sort 函数是系统定义好的,可以直接支持函数地址作为参数

bool cmp(int x,int y){
    return x > y;
    //注意: 此处不可出现等号'='
    //意思是:如果条件 x > y 成立,则 x 排在 y 前面 
}
int main(){
    ...
    sort (arr , arr + n, cmp);
    ... 
} 

 

例题:整数排序

 

 

 

#include<stdio.h>
#include<algorithm>
using namespace std;
int s[105];
bool cmp(int x,int y){
    if(x % 3 != y  % 3){
        return x % 3 < y % 3;
    }else{
        return x < y;
    }
}
int main(){
    freopen("lab.txt","r",stdin);
    int n;
    scanf("%d",&n);
    for(int i = 0; i < n;i++){
        scanf("%d",&s[i]);
    }
    sort(s,s + n,cmp);
    printf("%d",s[0]);
    for(int i = 1;i < n;i++){
        printf(" %d",s[i]);
    }
    return 0;
}

三、结构体的构造函数

 构造函数是一种定义在结构体中的特殊函数,它可以被用于描述对结构体进行初始化的算法。

struct Student{
    int score;
    string name;
    Student(string n,int s){
        name = n;
        score = s;
    }
};

●函数名与结构体名完全相同
●不能定义返回值类型,也不能有return语句
●可以有形参,也可以没有形参,可以带有默认参数
●可以重载
在使用的时候,我们不需要手动调用构造函数。当我们创建结构体的时候,构造函数会自动被调用

 

 

 

 

 

 

 

 

 

 

 

 

练习:读入多个学生姓名和成绩,反转输出成绩和姓名

#include<iostream>
#include<string>
using namespace std;

struct Student{
    string name;
    int score;
    Student(){}
    Student(string n,int s):name(n),score(s){}
};
int main(){
    Student stu[3];
    for(int i = 0;i < 3;i++){
        int s;
        string n;
        cin>>n>>s;
        stu[i] = Student(n,s);
    }
    for(int i = 0;i < 3;i++){
        cout<< stu[i].score<<" "<<stu[i].name<<endl;
    }
    return 0;
}

练习:输出成绩单,按照姓名的字典序排序 ———— 结构体排序

#include<stdio.h>
#include<string>
#include<iostream>
#include<algorithm>
using namespace std;
struct Student{
    string name;
    int score[2];
};
bool cmp(Student x,Student y){
    return x.name < y.name;
}
int main(){
    Student stu[4];
    for(int i =0; i < 4;i++){
        cin >> stu[i].name;
        for(int j = 0;j < 2;j++){
            cin >> stu[i].score[j];
        }
    }
    sort(stu,stu+4,cmp);
    for(int i = 0;i < 4;i++){
        cout<<stu[i].name<<" ";
        for(int j = 0; j < 2;j++){
            cout<<stu[i].score[j]<<" ";
        }
        cout<<endl;
    }
    return 0;
}

练习:复杂排序

 

 

其实就是改一下cmp 函数

bool cmp(Student x,Student y){
    if(x.score[0] != y.score[0])
        return x.score[0] > y.score[0];
    if(x.score[1] != y.score[1])
        return x.score[1] > y.score[0];
    if(x.score[2] != y.score[2])
        return x.score[2] > y.score[2];
    return x.score[3] > y.score[3];
}

 

例题:评奖

 

 

 这道题,可以按照上方结构体排序做

但有个更方便的做法:

#include<stdio.h>
int sum[805] = {0};  
char name[45][15];

int main(){
    int n;
    scanf("%d",&n);
    for(int i = 1; i <= n;i++){
        scanf("%s",name[i]);
        int all = 0;
        for(int j = 0; j < 4;j++){
            int k;
            scanf("%d",&k);
            all += k;
        }
        sum[all] = i;  
    }
    int f = 3;
    for(int i = 804; f > 0;i--){
        if(sum[i] != 0){
            f--;
            printf("%s\n",name[sum[i]]);
        }
    }
    return 0;
}

就是利用标记数组,其中每科最大分数为200,所以总分最大为800 。标记数组中放置总分对应的名字在name数组中的下标 ,然后从后往前遍历得到3个下标,分别输出相应的名字即可

posted @ 2020-02-15 21:48  远征i  阅读(680)  评论(0编辑  收藏  举报