快速排序能否在O(nLogn)最坏情况下实现?

一个典型的 快速排序 是O(n) 2. )。最坏的情况发生在拾取的轴始终是极端(最小或最大)元素时。当对输入数组进行排序或反向排序,并选择第一个或最后一个元素作为轴时,就会发生这种情况。

null

尽管即使在对数组进行排序时,随机快速排序也能很好地工作,但仍有可能随机选取的元素总是极端的。最坏的情况是否可以简化为O(nLogn)?

答案是肯定的,我们可以在最坏的情况下实现O(nLogn)。这个想法基于这样一个事实: 未排序数组的中值元素可以在线性时间内找到 .所以我们先找到中位数,然后围绕中位数元素划分数组。 下面是基于上述思想的C++实现。下面程序中的大多数函数都是从 未排序数组中第K个最小/最大元素|集3(最坏情况线性时间)

C++

/* A worst case O(nLogn) implementation of quicksort */
#include<cstring>
#include<iostream>
#include<algorithm>
#include<climits>
using namespace std;
// Following functions are taken from http://goo.gl/ih05BF
int partition( int arr[], int l, int r, int k);
int kthSmallest( int arr[], int l, int r, int k);
/* A O(nLogn) time complexity function for sorting arr[l..h] */
void quickSort( int arr[], int l, int h)
{
if (l < h)
{
// Find size of current subarray
int n = h-l+1;
// Find median of arr[].
int med = kthSmallest(arr, l, h, n/2);
// Partition the array around median
int p = partition(arr, l, h, med);
// Recur for left and right of partition
quickSort(arr, l, p - 1);
quickSort(arr, p + 1, h);
}
}
// A simple function to find median of arr[].  This is called
// only for an array of size 5 in this program.
int findMedian( int arr[], int n)
{
sort(arr, arr+n); // Sort the array
return arr[n/2]; // Return middle element
}
// Returns k'th smallest element in arr[l..r] in worst case
// linear time. ASSUMPTION: ALL ELEMENTS IN ARR[] ARE DISTINCT
int kthSmallest( int arr[], int l, int r, int k)
{
// If k is smaller than number of elements in array
if (k > 0 && k <= r - l + 1)
{
int n = r-l+1; // Number of elements in arr[l..r]
// Divide arr[] in groups of size 5, calculate median
// of every group and store it in median[] array.
int i, median[(n+4)/5]; // There will be floor((n+4)/5) groups;
for (i=0; i<n/5; i++)
median[i] = findMedian(arr+l+i*5, 5);
if (i*5 < n) //For last group with less than 5 elements
{
median[i] = findMedian(arr+l+i*5, n%5);
i++;
}
// Find median of all medians using recursive call.
// If median[] has only one element, then no need
// of recursive call
int medOfMed = (i == 1)? median[i-1]:
kthSmallest(median, 0, i-1, i/2);
// Partition the array around a random element and
// get position of pivot element in sorted array
int pos = partition(arr, l, r, medOfMed);
// If position is same as k
if (pos-l == k-1)
return arr[pos];
if (pos-l > k-1) // If position is more, recur for left
return kthSmallest(arr, l, pos-1, k);
// Else recur for right subarray
return kthSmallest(arr, pos+1, r, k-pos+l-1);
}
// If k is more than number of elements in array
return INT_MAX;
}
void swap( int *a, int *b)
{
int temp = *a;
*a = *b;
*b = temp;
}
// It searches for x in arr[l..r], and partitions the array
// around x.
int partition( int arr[], int l, int r, int x)
{
// Search for x in arr[l..r] and move it to end
int i;
for (i=l; i<r; i++)
if (arr[i] == x)
break ;
swap(&arr[i], &arr[r]);
// Standard partition algorithm
i = l;
for ( int j = l; j <= r - 1; j++)
{
if (arr[j] <= x)
{
swap(&arr[i], &arr[j]);
i++;
}
}
swap(&arr[i], &arr[r]);
return i;
}
/* Function to print an array */
void printArray( int arr[], int size)
{
int i;
for (i=0; i < size; i++)
cout << arr[i] << " " ;
cout << endl;
}
// Driver program to test above functions
int main()
{
int arr[] = {1000, 10, 7, 8, 9, 30, 900, 1, 5, 6, 20};
int n = sizeof (arr)/ sizeof (arr[0]);
quickSort(arr, 0, n-1);
cout << "Sorted array is" ;
printArray(arr, n);
return 0;
}


输出:

Sorted array is1 5 6 7 8 9 10 20 30 900 1000

快速排序在实践中是如何实现的?是否使用了上述方法? 尽管上述方法的最坏情况时间复杂度为O(nLogn),但它从未在实际实现中使用。与普通快速排序相比,这种方法中的隐藏常数较高。以下是QuickSort实际实现中使用的一些技术。 1) 随机挑选以减少最坏情况发生的可能性(随机快速排序) 2) 为小型数组调用插入排序以减少递归调用。 3) 快速排序是 尾部递归 ,这样就完成了尾部调用优化。

因此,上面讨论的方法更多地是一种理论方法,具有O(nLogn)最坏情况时间复杂度。

本文由 希瓦姆 。如果您发现任何不正确的地方,或者您想分享有关上述主题的更多信息,请发表评论

© 版权声明
THE END
喜欢就支持一下吧
点赞7 分享