【华为OD题库-005】选修课-Java

题目

现有两门选修课,每门选修课都有一部分学生选修,每个学生都有选修课的成绩,需要你找出同时选修了两门选修课的学生,先按照班级进行划分,班级编号小的先输出,每个班级按照两门选修课成绩和的降序排序,成绩相同时按照学生的学号升序排序。
输入描述
第一行为第—门选修课学生的成绩
第二行为第二门选修课学生的成绩,
每行数据中学生之间以英文分号分隔,每个学生的学号和成绩以英文逗号分隔,学生学号的格式为8位数字(2位院系编号+入学年份后2位+院系内部1位专业编号+所在班级3位学号),学生成绩的取值范围为[0,100]之间的整数,两门选修课选修学生数的取值范围为[1-2000]之间的整数。
输出描述
同时选修了两门选修课的学生的学号,如果没有同时选修两门选修课的学生输出NULL,否则,先按照班级划分,班级编号小的先输出,每个班级先输出班级编号(学号前五位),然后另起一行输出这个班级同时选修两门选修课的学生学号,学号按照要求排序(按照两门选修课成绩和的降序,成绩和相同时按照学号升序),学生之间以英文分号分隔。
示例1
输入:
01202021,75;01201033,95;01202008,80;01203006,90;01203088,100
01202008,70;01203088,85;01202111,80;01202021,75;01201100,88
输出:
01202
01202008;01202021
01203
01203088
说明:
同时选修了两门选修课的学生01202021、01202008、01203088,这三个学生两门选修课的成绩和分别为150、150、185。01202021、01202008属于01202班的学生,按照成绩和降序,成绩相同时按学号升序输出的结果为:01202008:01202021;01203088属于01203班的学生,按照成绩和降序,成绩相同时按学号升序输出的结果为01203088,01202的班级编号小于01203的班级编号,需要先输出。
示例2
输入:
01201022,75;01202033,95;01202018,80;01203006,90;01202066,100
01202008,70;01203102,85;01202111,80;01201021,75;01201100,88
输出:
NULL
说明:
没有同时选修了两门选修课的学生,输出NULL。

思路

题目描述多,实际很简单。
用两个list分别记录两门课程对应的学生编号,两个list相交的部分就能得到“同时选修了两门选修课的学生的学号”。
关键在于排序,题目要求:先按班级升序,再按成绩降序,最后按学号升序排。班级和学号都可以从学生编号中提取,此处的成绩指的两门课的总成绩,所以可以使用一个map来根据学号查询总成绩。

题解

package hwod;

import java.util.*;

public class ElectiveCourse {

    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        Map<String, Integer> map = new HashMap<>(); //key为学号,value为总分
        List<String> firstIds = new ArrayList<>();
        List<String> secondIds = new ArrayList<>();
        for (int i = 0; i < 2; i++) {
            String line = sc.nextLine();
            for (String item : line.split(";")) {
                String sid = item.split(",")[0];
                String score = item.split(",")[1];
                map.put(sid, map.getOrDefault(sid, 0) + Integer.valueOf(score));
                if (i == 0) firstIds.add(sid);
                else secondIds.add(sid);
            }

        }
        List<String> out_ids = new ArrayList<>(); //同时选修了两门课程的学生学号
        for (int i = 0; i < firstIds.size(); i++) {
            if (secondIds.contains(firstIds.get(i))) {
                out_ids.add(firstIds.get(i));
            }
        }
        if (out_ids.size() == 0) {
            System.out.println("NULL");
            return;
        }
        //学号按照班级,成绩,学号排序
        Collections.sort(out_ids, (A, B) -> {
            if (!A.substring(0, 5).equals(B.substring(0, 5))) {
                return A.substring(0, 5).compareTo(B.substring(0, 5));
            }
            if (!map.get(A).equals(map.get(B))) {
                return -map.get(A).compareTo(map.get(B));
            }
            return A.compareTo(B);
        });

        //打印输出
        String classid = null;
        for (int i = 0; i < out_ids.size(); i++) {
            boolean isNewClass = false;
            if (classid == null || !classid.equals(out_ids.get(i).substring(0, 5))) {
                classid = out_ids.get(i).substring(0, 5);
                if(i!=0) System.out.println();
                System.out.println(classid);
                isNewClass = true;
            }
            if (isNewClass) System.out.print(out_ids.get(i));
            else System.out.print(";" + out_ids.get(i));


        }
    }

}

推荐

如果你对本系列的其他题目感兴趣,可以参考华为OD机试真题及题解(JAVA),查看当前专栏更新的所有题目。

说明

本专栏所有文章均为原创,欢迎转载,请注明文章出处:https://blog.csdn.net/qq_31076523/article/details/134176793。百度和各类采集站皆不可信,搜索请谨慎鉴别。技术类文章一般都有时效性,本人习惯不定期对自己的博文进行修正和更新,因此请访问出处以查看本文的最新版本。

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
华为od机试中,新学校选址问题是一个关于使用Java语言解决的问题。在解决这个问题时,我们可以通过以下步骤来完成。 首先,我们需要理解问题的要求和限制条件。新学校选址的目标是在一个给定的地图上找到一个合适的位置来建设新学校。这个位置应满足一定的条件,比如与周围的住宅区距离不宜过远,应尽可能靠近居民区。限制条件可能还包括学校面积和周边环境等。 接下来,我们需要获取地图的数据。可以通过读取一个地图文件或者从数据库中查询地图数据来获得地图的信息。地图数据的存储方式可以灵活选择,比如使用二维数组或者邻接矩阵。 然后,我们需要编写一个Java程序来实现新学校选址算法。可以使用图的遍历算法,比如深度优先搜索(DFS)或广度优先搜索(BFS)来实现。算法的核心是通过递归或循环遍历地图上的每一个位置,并根据选址条件进行判断和筛选。 最后,我们需要输出选址结果。可以将选定的位置以某种方式标记在地图上,比如输出一个新的地图文件或者在图形界面中显示。同时,还可以输出选址的具体坐标和其他相关信息,以便后续的学校建设工作。 总之,通过使用Java语言,我们可以通过一个新学校选址问题来展示我们在算法设计和编程方面的能力。相信在华为od机试中,通过合理的数据处理和算法实现,我们可以解决这个问题并得出满意的选址结果。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值