在昨天参加了东哥的笔试,选择题做的还算可以,但是还有道编程题和关于jdk8的Stream特性难住了。鉴于此用博客总结一下这道编程题,并结合Stream特性来简化代码,熟悉Api。
题目描述
某校在积极推行无人监考制度,但是总有学生是不自觉的,如果将两个很熟的异性朋友放在同一个考场里,他们就会交流甚至作弊。因此一个考场中不能允许两个很熟的异性朋友存在,学校希望通过搬出一部分学生的方法来改善这一问题。但是又因为教室数量有限,因此希望一个教室中容下的学生尽可能多,即需要搬出教室的学生数量尽可能少,请你输出搬出教室人数最少,且字典序最小的方案。
输入
输入第一行有两个整数n和m,分别表示有n个男生和n个女生,有m个朋友关系。(1<=n<=500,1<=m<=100000)接下来m行,每行有两个整数,x和y,表示第x号男生和第y号女生是朋友。
男生的编号均为[1,n],女生的编号为[n+1,2n]。
输出
输出第一行包含一个整数a,表示最少需要搬出教室的人数。
输出第二行有a个整数,即a个需要搬出教室的人的编号,要求人数最少,且字典序最小。
意思是:一个教室内不能有亲密关系的男女,选择要移除的人满足尽可能的少、学号尽可能的小。
思路
按照题目描述可以明确:移除人数最少代表先移除不止和一个人有关系的同学。学号尽可能的小代表着优先移除男生。它的优先级是:人数>学号。
有了这个目标,再来确定算法的数据结构,我运用了以下的数据结构:
1 //维护每个学生的关系映射,一对多的关系 2 Map<Integer, List<Integer>> relationmap = new HashMap<>(16); 3 4 //某个学生的关系具体学号 5 List templist = relationmap.getOrDefault(boynum, new ArrayList<>()); 6 7 //存放男女生有关系的个数,男生与女生各n个,第一位不放数据。 8 int[] relation = new int[2*n+1];
然后利用伪代码来具体描述这个过程:
获得输入的男生人数n和关系m; 初始化relation数组,map; for( m个关系){ 接收男生与女生的关系; 将两个学号作key放入Map中; 把另一个学号作value放入map的List中; 以两个学号为下标的relation数组加一; } while(true){ 从头到尾遍历relation数组,获得最大值的下标maxRelationIndex; 当maxRelationIndex==0,意味着当前没有亲密关系,break; 根据maxRelationIndex获得map中的关系结合list; for(list){ 遍历list,将对应的relation数组的下标-1; } 删除以学号为索引在relation数组的记录; 记录被删除的学号。 } 输出记录:
代码实现
1 import java.util.*; 2 3 /** 4 * TODO 5 * @Author: HILL 6 * @date: 2019/8/25 9:50 7 * 8 **/ 9 public class Main { 10 public static void main(String[] args) { 11 12 //维护关系映射 13 Map<Integer, List<Integer>> relationmap = new HashMap<>(16); 14 //男女生人数 15 Scanner sc = new Scanner(System.in); 16 int n = sc.nextInt(); 17 int m = sc.nextInt(); 18 19 //存放男女生有关系的个数 20 int[] relation = new int[2*n+1]; 21 22 23 for (int i=0;i<m;i++){ 24 int boynum = sc.nextInt(); 25 int girlnum = sc.nextInt(); 26 27 //添加男生关系映射 28 List templist = relationmap.getOrDefault(boynum, new ArrayList<>()); 29 templist.add(girlnum); 30 relationmap.put(boynum,templist); 31 //添加女生关系映射 32 templist = relationmap.getOrDefault(girlnum, new ArrayList<>()); 33 templist.add(boynum); 34 relationmap.put(girlnum,templist); 35 //维护每个人的关系度的权值,越大代表与越多人有关系 36 relation[girlnum]++; 37 relation[boynum]++; 38 39 40 } 41 List<Integer> result = new ArrayList<>(); 42 while (true){ 43 int maxRelationIndex = 0; 44 45 //从头到尾遍历,优先移除男生 46 for (int i=1 ;i<relation.length;i++ ){ 47 if (relation[i]>maxRelationIndex){ 48 maxRelationIndex = i; 49 } 50 } 51 //当教室里没有亲密关系时 52 if (maxRelationIndex == 0){ 53 break; 54 } 55 56 //优先移除与最多人有关系的学生 57 relation[maxRelationIndex] = 0; 58 //查出所有与被移除学生有关系的学生 59 List<Integer> list = relationmap.get(maxRelationIndex); 60 //将它们的关系计数-1 61 list.forEach(i-> relation[i]--); 62 63 //将移除的学生加入到结果集 64 result.add(maxRelationIndex); 65 relationmap.remove(maxRelationIndex); 66 } 67 System.out.println(result.size()); 68 result.forEach(num->System.out.print(num+" ")); 69 70 } 71 }
提醒
以上代码仅供交流参考,用例不一定全部通过,因为当我想出来时已经没时间了。从总体上分析,性能估计不会很高,毕竟一个算法下来会有两次循环,同时空间复杂度也比较高。但能做一步是一步嘛,后面再看看能不能换个思路。