package practice; import edu.princeton.cs.algs4.*; public class TestMain {
public static void main(String[] args) {
int[] a = new int[20];
for (int i = 0; i < a.length; i++) {
int temp = (int)(StdRandom.uniform(1, 10));
a[i] = temp;
} for (int i : a) {
System.out.print(i+" ");
}
System.out.println();
ToSort.sort3Way(a);
for (int i : a) {
System.out.print(i+" ");
}
}
} class ToSort{
private static int[] anx;
/*
* 我第一次写出来的快速排序,容易理解,但需要一倍的额外空间,也不快.....
*/
public static void mySort(int[] a) {
anx = new int[a.length];
myQuickSort(a, 0, a.length - 1);
}
public static void myQuickSort(int[] a,int lo,int hi) {
if (lo >= hi) return;
int m = lo;
int n = hi;
int mid = a[lo];
for (int i = lo + 1; i <= hi; i++) {
if (a[i] <= mid) anx[m++] = a[i];
else if (a[i] > mid) anx[n--] = a[i];
}
//此处往后m,n相等
anx[m] = mid;
for (int i = lo; i <= hi; i++) a[i] = anx[i];
myQuickSort(a, lo, m - 1);
myQuickSort(a, n + 1, hi);
}
/*
* 快速排序 时间复杂度O(NlgN)
*/
public static void sort(int[] a) {
quickSort(a, 0, a.length - 1);
}
public static void quickSort(int[] a,int lo,int hi) {
if (lo >= hi) return; //如果分到只剩一个元素,或没有元素,则返回
int n = partition(a, lo, hi); //将数组切分,取一个中值,分成比它小的一部分和比它大的一部分
quickSort(a, lo, n - 1); //将两部分分别快速排序
quickSort(a, n + 1, hi);
}
public static int partition(int[] a,int lo,int hi) {
int mid = a[lo]; //把第一个元素设为中值,先放在最左边不管
int m = lo;
int n = hi + 1;
while (true) {
while (a[++m] < mid) if(m == hi) break; //从左开始找出一个比mid大的元素
while (a[--n] > mid)/*if(n == lo) break; 这句没用,但可以让人更容易理解程序*/;//从右开始找出一个比mid小的元素
if (m >= n) break; //找完了
exch(a, m, n);
}
exch(a, n, lo); //把最左端中值和"比它小的值中最右端的数"换位置,它就到最中间了
return n;
}
/*
* 三向切分的快速排序,适用于有大量重复元素的数组
*/
public static void sort3Way(int[] a) {
quickSort(a, 0, a.length - 1);
}
public static void quickSort3Way(int[] a,int lo,int hi) {
if (lo >= hi) return;
int mid = a[lo];
int lt = lo + 1;
int gt = hi;
int i = lo + 1;
while (i <= gt) { //将比中值小的放在最左端,比中值大的放在最右端,和中值相等的放在中间
if (a[i] < mid) exch(a, lt++, i++); //换过来的值还是与中值相等的值,所以不用动
else if (a[i] > mid) exch(a, gt--, i); //换过来的值不知道是什么值,所以还要处理
else if (a[i] == mid) i++; //相等就过
}
exch(a, lo, lt - 1); //把最左端中值和"比它小的值中最右端的数"换位置
quickSort3Way(a, lo, lt - 2); //调用自己处理剩下的两部分
quickSort3Way(a, gt + 1, hi);
}
/*
* 交换a[i]与a[j]的值
*/
private static void exch(int[] a, int i, int j) {
int t = a[i];
a[i] = a[j];
a[j] = t;
}
}
快速排序示意图(图片来自《算法(第四版官网)》)
三向切分的快速排序示意图(图片来自《算法(第四版官网)》)