48天强训 & Day1 & JavaOj
1. 编程题1 - 组队竞赛
组队竞赛_牛客笔试题_牛客网 (nowcoder.com)
1.1 读题
1.2 算法思想基础
- 我们应该尽量的让每一个队伍的中间值都最大化~
- 我们应该尽量的让每一个队伍的最小值都足够小~
- 前33%的不应该都作为每个队伍的最大值~
- 接下来我将讲解每个位置应该如何选组员~
- 但是这里我要说一个前提
- 首先,我们需要把所有人进行一个排序~
- 分为前三分之一,中三分之一,后三分之一
- 后三分之一为整体水平排名靠后的~
1.2.1 后三分之一
- 我们要想让一个队伍的最小值足够小
- 那么我们可以让那么就选整体水平的后三分之一
- 事实也是如此~
证明:(反证法)
- 首先:
- 其次:
- 得出结论:
- 也就是说,为了让组队水平最大化,我们应该让每个队伍的最小值,要在整体的后三分之一里去选~
- 并且这么选都可以,因为最大值和中间值都会比最小值大,并且队伍的水平不受最小值影响~
1.2.2 前三分之二
- 我们确定了每个队伍的最小值后,紧接着就要确定一个一个的队伍了~
- 切记,并不能让前三分之一的人都作为每个队伍的最大值,这样非常影响总体水平~
- 原理跟刚才差不多,可以用反证法证明~
- 其实这样排是有可能做对的
- 但是,肯定是有漏洞的~
- 而正确的思想是,一个一个队伍的去确定,每次确定都让这个队伍水平最大化~
- 对于第二个队伍
- 以此类推~
- 所以,最大水平组队方式就是这样的
- 水平总和最大值为:
- 假设有n个队伍
- 所有人从大到小排为 a3n
- 水平总和值为Sn
- 则Snmax = a2 + a4 + a6 + ······ + a2n
1.3 代码设计
- n为队伍的个数
- 那么我们只需要知道前2n名
- topK问题
- 用优先级队列 - 堆
- 前k大,用小根堆
- 每次去梢(poll),就是去掉最小值
- 这里就相当于去掉a2n、a2n-1 ······
- 按照上面的算法,我们可以从a2n + a2n-2 + ······ + a4 + a2
- 即从后往前加
- 当然也可以直接用各种排序方式去排序,然后按照下标依次相加~
public class Main {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
while(scanner.hasNextInt()) {
int number = scanner.nextInt();
PriorityQueue<Integer> priorityQueue = new PriorityQueue<>();
for (int i = 0; i < number * 3; i++) {
int value = scanner.nextInt();
if(i < 2 * number) {
priorityQueue.offer(value);
}else {
if(value > priorityQueue.peek()) {
priorityQueue.poll();
priorityQueue.offer(value);
}
}
}
long result = 0;
for (int i = 0; i < number; i++) {
result += priorityQueue.poll();
priorityQueue.poll();
}
System.out.println(result);
}
}
}
- 解析: