在昨天参加了东哥的笔试,选择题做的还算可以,但是还有道编程题和关于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> 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> 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
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 result = new ArrayList<>();
42 while (true){
43 int maxRelationIndex = 0;
44
45 //从头到尾遍历,优先移除男生
46 for (int i=1 ;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 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 }
提醒
以上代码仅供交流参考,用例不一定全部通过,因为当我想出来时已经没时间了。从总体上分析,性能估计不会很高,毕竟一个算法下来会有两次循环,同时空间复杂度也比较高。但能做一步是一步嘛,后面再看看能不能换个思路。