NOIP2007奖学金问题及简单背包问题

2024-03-30 18:20

本文主要是介绍NOIP2007奖学金问题及简单背包问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

引言:

最近刷了一些洛谷的基础算法题,关于排序类和贪心类,也发现了结构体用在这两种题目上的异曲同工之妙,那么我们废话不多说,直接上题目与解析帮助大家更好地掌握。

题目描述

某小学最近得到了一笔赞助,打算拿出其中一部分为学习成绩优秀的前 55 名学生发奖学金。期末,每个学生都有 33 门课的成绩:语文、数学、英语。先按总分从高到低排序,如果两个同学总分相同,再按语文成绩从高到低排序,如果两个同学总分和语文成绩都相同,那么规定学号小的同学排在前面,这样,每个学生的排序是唯一确定的。

任务:先根据输入的 33 门课的成绩计算总分,然后按上述规则排序,最后按排名顺序输出前五名名学生的学号和总分。

注意,在前 55 名同学中,每个人的奖学金都不相同,因此,你必须严格按上述规则排序。例如,在某个正确答案中,如果前两行的输出数据(每行输出两个数:学号、总分) 是:

7 279  
5 279

这两行数据的含义是:总分最高的两个同学的学号依次是 77 号、55 号。这两名同学的总分都是 279279 (总分等于输入的语文、数学、英语三科成绩之和) ,但学号为 77 的学生语文成绩更高一些。

如果你的前两名的输出数据是:

5 279  
7 279

则按输出错误处理,不能得分。

输入格式

共 n+1 行。

第 1 行为一个正整数 n≤300,表示该校参加评选的学生人数。

第 2 到 n+1 行,每行有 3 个用空格隔开的数字,每个数字都在 0 到 100 之间。第 j 行的 3 个数字依次表示学号为 j−1 的学生的语文、数学、英语的成绩。每个学生的学号按照输入顺序编号为 1∼n(恰好是输入数据的行号减 1)。

保证所给的数据都是正确的,不必检验。

输出格式

共 5行,每行是两个用空格隔开的正整数,依次表示前 5 名学生的学号和总分。

输入输出样例

输入 #1复制

6
90 67 80
87 66 91
78 89 91
88 99 77
67 89 64
78 89 98

输出 #1复制

6 265
4 264
3 258
2 244
1 237

输入 #2复制

8
80 89 89
88 98 78
90 67 80
87 66 91
78 89 91
88 99 77
67 89 64
78 89 98

输出 #2复制

8 265
2 264
6 264
1 258
5 258

我们发现,排序的优先级是总分大的优先,总分相同则比较语文成绩,语文成绩相同则比较学号,学号小的同学排前面,这里需要注意,前面都是大的在前,唯有学号是小的在前,这是等下要编写的一个部分。

那我们来初步实现一下代码块:

代码块实现:

struct stu
{int num;int a,b,c;int sum;
}student[100005];

我们定义一个结构体部分负责接收学号,三科目成绩,总成绩等信息。

bool cmp(stu a,stu b)
{if(a.sum>b.sum){return 1;}else if(a.sum<b.sum){return 0;}else{if(a.y>b.y){return 1;}else if(a.y<b.y){return 0;}else{if(a.num<b.num){return 1;}else{return 0;}}}}

这里我们定义一个bool类型的比较函数,方便后续的排序中sort使用,我们依据推理逻辑,优先总分在先,再次语文,最后学号,注意是哪种情况该先排,哪种情况又该后排。

int main()
{int n;cin >> n;for(int i=1;i<=n;i++){student[i].num=i;cin >> student[i].yu >> student[i].shu >> student[i].eng;student[i].sum=student[i].yu+student[i].shu+student[i].eng;}sort(student+1,student+1+n,cmp);for(int i=1;i<=5;i++){cout << student[i].num << " " << student[i].sum << endl;}return 0;

这里是主函数的部分,注意结构体传数字时的操作,并使用sort根据上面的逻辑排序,最后输出即大功告成,这一题主要可以锻炼大家对于结构体与排序方法的掌握,很建议大家去做一下,最后完整代码和题目链接奉上:

#include <iostream>
#include <algorithm>
using namespace std;
struct stu
{int num;int yu,shu,eng;int sum;
}student[10005];
bool cmp(stu a,stu b)
{if(a.sum>b.sum){return 1;}else if(a.sum<b.sum){return 0;}else{if(a.yu>b.yu){return 1;}else if(a.yu<b.yu){return 0;}else{if(a.num>b.num){return 0;}else{return 1;}}}
}
int main()
{int n;cin >> n;for(int i=1;i<=n;i++){student[i].num=i;cin >> student[i].yu >> student[i].shu >> student[i].eng;student[i].sum=student[i].yu+student[i].shu+student[i].eng;}sort(student+1,student+1+n,cmp);for(int i=1;i<=5;i++){cout << student[i].num << " " << student[i].sum << endl;}return 0;
}

题目链接:https://www.luogu.com.cn/problem/P1093

部分背包问题:

在掌握了上一题之后,面对一部分的背包问题,我们是不是也可以使用异曲同工的方法来做出来呢?看下面这一题:

题目描述

阿里巴巴走进了装满宝藏的藏宝洞。藏宝洞里面有 N(N≤100) 堆金币,第 i 堆金币的总重量和总价值分别是 mi(1≤mi≤100),vi​​(1≤vi​≤100)。阿里巴巴有一个承重量为 (T≤1000) 的背包,但并不一定有办法将全部的金币都装进去。他想装走尽可能多价值的金币。所有金币都可以随意分割,分割完的金币重量价值比(也就是单位价格)不变。请问阿里巴巴最多可以拿走多少价值的金币?

输入格式

第一行两个整数 N,T。

接下来 N 行,每行两个整数 mi​,vi​。

输出格式

一个实数表示答案,输出两位小数

输入输出样例

输入 #1复制

4 50
10 60
20 100
30 120
15 45

输出 #1复制

240.00

首先,题目很明确了我们可以分割单块金币,以获得权重最大的金币(权重=价值/质量),我们肯定优先选择相同质量价值最高的金币堆,那么我们就可以对权重进行排序,再慢慢比较背包质量与金币质量,直到背包刚好装满为止。

结构体定义:

struct value
{int w,v;double jun;
}money[100005];

再定义比较函数:

bool cmp(Value a,Value b)
{if(a.jun>b.jun){return 1;}else{return 0;}
}

主函数部分:

int main()
{int n,t;cin >> n >> t;for(int i=0;i<n;i++){cin >> money[i].w >> money[i].v;money[i].jun=money[i].v/money[i].w;}sort(money,money+n,cmp);double sum;for(int i=0;i<n;i++){if(t>money[i].w){sum+=money[i].v;t-=money[i].w;}else{sum+=t*money[i].jun;break;}}printf("%.2lf",sum);return 0;

这一题我觉得主函数部分是难于自定义函数部分的操作的,还有最后这一个break操作,一定要记得加上,也就是背包最后装块的部分,装了以后背包就满了。最后,完整代码和题目地址奉上:

#include <iostream>
#include <algorithm> 
using namespace std;
struct Value
{double w,v;double jun;
}money[10005];
bool cmp(Value a,Value b)
{if(a.jun>b.jun){return 1;}else{return 0;}
}
int main()
{int n,t;cin >> n >> t;for(int i=0;i<n;i++){cin >> money[i].w >> money[i].v;money[i].jun=money[i].v/money[i].w;}sort(money,money+n,cmp);double sum;for(int i=0;i<n;i++){if(t>money[i].w){sum+=money[i].v;t-=money[i].w;}else{sum+=t*money[i].jun;break;}}printf("%.2lf",sum);return 0;
}

题目链接:https://www.luogu.com.cn/problem/P2240

总结:

今天给大家介绍了两个题目,它们的布局大体相同,主要是为了大家能够掌握使用结构体,排序,自定义函数的结合使用来解决问题,制作不易,给个点赞鼓励一下吧!

这篇关于NOIP2007奖学金问题及简单背包问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/862231

相关文章

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常

Vue项目中Element UI组件未注册的问题原因及解决方法

《Vue项目中ElementUI组件未注册的问题原因及解决方法》在Vue项目中使用ElementUI组件库时,开发者可能会遇到一些常见问题,例如组件未正确注册导致的警告或错误,本文将详细探讨这些问题... 目录引言一、问题背景1.1 错误信息分析1.2 问题原因二、解决方法2.1 全局引入 Element

关于@MapperScan和@ComponentScan的使用问题

《关于@MapperScan和@ComponentScan的使用问题》文章介绍了在使用`@MapperScan`和`@ComponentScan`时可能会遇到的包扫描冲突问题,并提供了解决方法,同时,... 目录@MapperScan和@ComponentScan的使用问题报错如下原因解决办法课外拓展总结@

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

利用Python编写一个简单的聊天机器人

《利用Python编写一个简单的聊天机器人》这篇文章主要为大家详细介绍了如何利用Python编写一个简单的聊天机器人,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 使用 python 编写一个简单的聊天机器人可以从最基础的逻辑开始,然后逐步加入更复杂的功能。这里我们将先实现一个简单的

numpy求解线性代数相关问题

《numpy求解线性代数相关问题》本文主要介绍了numpy求解线性代数相关问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 在numpy中有numpy.array类型和numpy.mat类型,前者是数组类型,后者是矩阵类型。数组

解决systemctl reload nginx重启Nginx服务报错:Job for nginx.service invalid问题

《解决systemctlreloadnginx重启Nginx服务报错:Jobfornginx.serviceinvalid问题》文章描述了通过`systemctlstatusnginx.se... 目录systemctl reload nginx重启Nginx服务报错:Job for nginx.javas

Redis缓存问题与缓存更新机制详解

《Redis缓存问题与缓存更新机制详解》本文主要介绍了缓存问题及其解决方案,包括缓存穿透、缓存击穿、缓存雪崩等问题的成因以及相应的预防和解决方法,同时,还详细探讨了缓存更新机制,包括不同情况下的缓存更... 目录一、缓存问题1.1 缓存穿透1.1.1 问题来源1.1.2 解决方案1.2 缓存击穿1.2.1