Chinaunix首页 | 论坛 | 博客
  • 博客访问: 220990
  • 博文数量: 136
  • 博客积分: 2919
  • 博客等级: 少校
  • 技术积分: 1299
  • 用 户 组: 普通用户
  • 注册时间: 2011-03-11 09:08
文章分类

全部博文(136)

文章存档

2013年(1)

2011年(135)

我的朋友

分类: C/C++

2011-03-15 09:12:25

2.2 Sorting

2011-03-03 Thu

  • One of the best all-round sorting algorithms is quicksort, which was invented in 1960 by C. A. R. Hoare. Quicksort is a fine example of how to avoid extra computing. It works by partitioning an array into little and big elements:
    1. pick one element of the array(the "pivot").
    2. partition the other elements into two groups:
    3.       "little ones" that are less than the pivot value, and
    4.       "big ones" that are greater than or equal to the pivot value.
    5. recursively sort each group.
  • When this process is finished, the array is in order.

  • Quicksort is fast because once an element is known to be less than the pivot value, we don't have to compare it to any of the big ones; similarly, big ones are not compared to little ones. This makes it much faster than the simple sorting methods such as insertion sort and bubble sort that compare each element directly to all the others.

  • This quicksort
  1. /* quicksort: sort v[0]..v[n-1] into increasing order */
  2. void quicksort(int v[], int n)
  3. {
  4.         int i, last;
  5.         if (n <= 1) /* nothing to do */
  6.                 return;
  7.         swap (v, 0, rand() % n); /* move pivot elem to v[0] */
  8.         last = 0;
  9.         for (i = 1; i < n; i++) /* partition */
  10.                 if (v[i] < v[0])
  11.                         swap(v, ++last, i);
  12.         swap(v, 0, last); /* restore pivot */
  13.         quicksort(v, last); /* recursively sort */
  14.         quicksort(v+last+1, n-last-1); /* each part */
  15. }

The swap operation, which interchanges two elements, appears three times in quicksoft, so it is best made into a separate function:

  1. /* swap: interchange v[i] and v[j] */
  2. void swap(int v[], int i, int j)
  3. {
  4.         int temp;
  5.         temp = v[i];
  6.         v[i] = v[j];
  7.         v[j] = temp;
  8. }

(1) Partitioning selects a random element as the pivot, swaps it temporarily to the front, then sweeps through the remaining elements, exchanging those smaller than the pivot ("little ones") towards the beginning (at location last) and big ones towards the end (at location i) At the beginning of the process, just after the pivot has been swapped to the front, last=0 and elements i=1 through n-1 are unexamined.

(2) At the top of the for loop, element 1 through last are strictly less than the pivot, elements last+1 through i-1 are greater than or equal to the pivot, and element i through n-1 have not been examined yet. Until v[i]>=v[0], the algorithm may swap v[i] with itself; this wastes some time but not enough to worry about.

(3) After all elements have been partitioned, element 0 is swapped with the last element to put the pivot element in its final position; this maintains the correct ordering.

(4) The same process is applied to the left and right sub-arrays; when this has finished, the whole array has been sorted.

How fast is quicksort? In the best possible case,

  • the first pass partitions n elements into two groups of about n/2 each.
  • the second level partitions two groups, each of about n/2 elements, into four groups each of about n/4.
  • the next level partitions four groups of about n/4 into eight of about n/8.
  • and so on.

This goes on for about log2 n levels, so the total amount of work in the best case is proportional to n+2× n/2+4× n/4+8× n/8 … (log2 n terms), which is nlog2 n. On the average, it does only little more work. It is customary to use base 2 logarithms; thus we say that quicksort takes time proportional to nlog n.

  • This implementation of quicksort is the clearest for exposition, but it has a weakness. If each choice of pivot splits the element values into two nearly equal groups, our analysis is correct, but if the split is uneven too often, the run-time can grow more like n2.
  • Our implementation uses a random element as the pivot to reduce the chance that unusual input data will cause too many uneven splits. But if all the input values are the same, our implementation splits off only one element each time and will thus run in time proportional to n2.
 
阅读(354) | 评论(0) | 转发(0) |
0

上一篇:tpop(2.1): Searching

下一篇:tpop(2.3): Libraries

给主人留下些什么吧!~~