题目名称 2929. [USACO Open18 Platinum] Out of Sorts
输入输出 sort.in/out
难度等级 ★★★
时间限制 1000 ms (1 s)
内存限制 256 MiB
测试数据 10
题目来源 GravatarWHZ0325 于2018-04-11加入
开放分组 全部用户
提交状态
分类标签
分享题解
通过:0, 提交:0, 通过率:0%
关于 Out of Sorts 的近10条评论(全部评论)
同上
Gravatar纸伞
2019-06-20 20:59 2楼
冒泡
Gravatar-1
2018-04-13 07:07 1楼

2929. [USACO Open18 Platinum] Out of Sorts

★★★   输入文件:sort.in   输出文件:sort.out   简单对比
时间限制:1 s   内存限制:256 MiB

【题目描述】

留意着农场之外的长期职业生涯的可能性,奶牛Bessie开始在不同的在线编程网站上学习算法。她最喜欢的两个算法是“冒泡排序”和“快速排序”,但是不幸的是Bessie轻易地把它们搞混了,最后实现了一个奇怪的混合算法!如果数组$A$中$A[...i]$的最大值不大于$A[i+1…]$的最小值,我们就称元素$i$和$i+1$之间的位置为一个“分隔点”。Bessie还记得快速排序包含对数组的重排,产生了一个分隔点,然后要递归对两侧的$A[...i]$和$A[i+1…]$排序。然而,尽管她正确地记下了数组中所有的分隔点都可以在线性时间内被求出,她却忘记快速排序应该怎么重排来快速构造一个分隔点了!在这个可能会被证明是排序算法的历史中最糟糕的算法性失误之下,她做出了一个不幸的决定,使用冒泡排序来完成这个任务。

以下是Bessie最初的对数组$A$进行排序的实现的概要。她首先写了一个简单的函数,执行冒泡排序的一轮:

bubble_sort_pass (A) {

    for i = 0 to length(A)-2

        if A[i] > A[i+1], swap A[i] and A[i+1]

}

她的快速排序(相当快)函数的递归代码是按下面的样子构成的:

quickish_sort (A) {

    if length(A) = 1, return

    do { // Main loop

        work_counter = work_counter + length(A)

        bubble_sort_pass(A)

    } while (no partition points exist in A)

    divide A at all partition points; recursively quickish_sort each piece

}

Bessie好奇于她的代码能够运行得多快。简单起见,她计算出她得主循环的每一轮都消耗线性时间,所以她相应增加一个全局变量work_counter的值,以此来跟踪整个算法总共完成的工作量。

给定一个输入数组,请预测quickish_sort函数接收这个数组之后,变量work_counter的最终值。

【输入格式】

输入的第一行包含$N$($1≤N≤100,000$)。接下来$N$行描述了$A[0]…A[N−1]$,每个数都是一个范围为$0…10^9$的整数。输入数据不保证各不相同。

【输出格式】

输出work_counter的最终值。

【输入样例】

7
20
2
3
4
9
8
7

【输出样例】

12

【提示】

在这个例子中,数组开始时为20 2 3 4 9 8 7。在一轮冒泡排序之后(增加7的工作量),我们得到2 | 3 | 4 | 9 8 7 | 20,其中|表示一个分隔点。于是我们的问题被分成了递归的子问题,包括对2、3、4、20排序(每个消耗0单元的工作量)和对9 8 7排序。对于9 8 7这个子问题,主循环的一轮(3单元工作量)得到8 7 | 9,在此之后最后一轮处理8 7(2单元工作量) 就有效地完成了排序。

【来源】

USACO Open18 Platinum Problem 1

供题:Brian Dean