/** * 將指定範圍的整形數組升序排序。 * x[] 待排數組 * off 從數組的第off個元素開始排序 * len 數組長度 */ private static void sort1(int x[], int off, int len) { //最佳化1:在小規模(size<7)數組中,直接插入排序的效率要比快速排序高。 if (len < 7) { for (int i=off; i<len+off; i++) for (int j=i; j>off && x[j-1]>x[j]; j--) swap(x, j, j-1); return; } //最佳化2:精心選擇劃分元素,即樞軸 //如果是小規模數組(size<=7),直接取中間元素作為樞軸 //如果是中等規模數組(7=<size<=40),則在數組首、中、尾三個位置上的數中取中間大小的數作為樞軸 //如果是大規模數組(size>40),則在9個指定的數中取一個偽中數(中間大小的數s) int m = off + (len >> 1); if (len > 7) { int l = off; int n = off + len - 1; if (len > 40) { int s = len/8; l = med3(x, l, l+s, l+2*s); m = med3(x, m-s, m, m+s); n = med3(x, n-2*s, n-s, n); } m = med3(x, l, m, n); } int v = x[m]; //最佳化3:每一次樞軸v的劃分,都會形成形成一個形如 (<v)* v* (>v)* //階段一,形成 v* (<v)* (>v)* v* 的數組 int a = off, b = a, c = off + len - 1, d = c; while(true) { while (b <= c && x[b] <= v) { if (x[b] == v) swap(x, a++, b); b++; } while (c >= b && x[c] >= v) { if (x[c] == v) swap(x, c, d--); c--; } if (b > c) break; swap(x, b++, c--); } //階段二,將樞軸和與樞軸相等的元素交換到數組中間 int s, n = off + len; s = Math.min(a-off, b-a ); vecswap(x, off, b-s, s); s = Math.min(d-c, n-d-1); vecswap(x, b, n-s, s); //階段三,遞迴排序與樞軸不相等都元素區間 if ((s = b-a) > 1) sort1(x, off, s); if ((s = d-c) > 1) sort1(x, n-s, s); }
|