// 复制数组 int * copyArray(int * arr, int n) { int i; int *newArr = (int *) malloc(sizeof(int) * n); for (i = 0; i < n; ++i) { newArr[i] = arr[i]; } return newArr; }
// 插入排序 voidInsertionSort(int * arr, int left, int right) { int i, j; for (i = left + 1; i <= right; ++i) { int temp = arr[i]; for (j = i; j > left && arr[j-1] > temp; j--) { arr[j] = arr[j - 1]; } arr[j] = temp; } }
// 打印数组内容 voidprintArray(int * arr, int n) { int i; for (i = 0; i < n; ++i) { printf("%d ", arr[i]); } printf("\n"); } #endif//ALGORITHM_SORTHELP_H
// 对arr[left...right]部分进行partition操作 // 返回posi,使得arr[left...p-1] < arr[p] ; arr[p+1...right] > arr[p] int __partition1(int *arr, int left, int right) { int i, j; /******************* 优化2 *******************/ swap(&arr[left], &arr[rand() % (right - left + 1) + left]);
int v = arr[left]; // 基准 j = left; for (i = left + 1; i <= right; ++i) { // 如果当前元素 比 基准小 if (arr[i] < v) { swap(&arr[i], &arr[j + 1]); j++; } // 比基准大 i++ } swap(&arr[left], &arr[j]); return j; }
// 对arr[left, right]进行快速排序 void __QuickSortOneWays1(int * arr, int left, int right) { /************ 优化1 **********/ if (right - left <= 15) // 递归结束条件 { InsertionSort(arr, left, right); return; } int p = __partition1(arr, left, right); // 排好序后 基准所在位置的索引 __QuickSortOneWays1(arr, left, p - 1); // 对左子序列进行快排 __QuickSortOneWays1(arr, p + 1, right); // 对右子序列进行快排 }
// 统一接口 voidQuickSortOneWays1(int * arr, int n) { srand(time(NULL)); __QuickSortOneWays1(arr, 0, n - 1); }
intmain(void) { // 生成随机数组测试 int n = 100000; int *arr = generateRandomArray(n, 0, n); // 生成n个[0,n]的随机数 testSort(arr, n, "QuickSortOneWays", QuickSortOneWays1);
// 生成 几乎有序的数组 最多只有2*100个元素无序 int swapTimes = 100; int *arr1 = generateNearlyOrderArray(n, swapTimes); testSort(arr1, n, "QuickSortOneWays", QuickSortOneWays1);
// 生成大量重复元素数组 [0,10] int *arr2 = generateRandomArray(n, 0, 10); testSort(arr2, n, "QuickSortOneWays",QuickSortOneWays1); return0; }
// arr[left+1...lt] < v ; arr[lt+1...i-1] == v ; arr[gt...right] >v void __QuickSortThreeWays(int *arr, int left, int right) { int i, lt, gt; if (right - left <= 15) // 递归结束条件 { InsertionSort(arr, left, right); return; }
swap(&arr[left], &arr[rand() % (right - left + 1) + left]); // 选取随机基准 与arr[left]交换 int v = arr[left]; // 基准
lt = left; // 将初始lt代入 [left+1...lt] < v 刚好使这个区间无效 gt = right + 1; // 同理 gt代入 [gt...right] >v i = left + 1; // 同理i代入 [lt+1...i-1] == v
// 统一接口 voidQuickSortThreeWays(int *arr, int n) { srand(time(NULL)); __QuickSortThreeWays(arr, 0, n - 1); }
intmain(void) { // 生成随机数组测试 int n = 1000000; int *arr = generateRandomArray(n, 0, n); // 生成n个[0,n]的随机数 testSort(arr, n, "QuickSortThreeWays", QuickSortThreeWays);
// 生成 几乎有序的数组 最多只有2*100个元素无序 int swapTimes = 100; int *arr1 = generateNearlyOrderArray(n, swapTimes); testSort(arr1, n, "QuickSortThreeWays", QuickSortThreeWays);
// 生成大量重复元素数组 [0,10] int *arr2 = generateRandomArray(n, 0, 10); testSort(arr2, n, "QuickSortThreeWays", QuickSortThreeWays);
intmain(void) { // 生成随机数组测试 int n = 1000000; printf("\ntest generateRandomArray, size = %d, random range [0, %d]\n ", n, n); int *arr1 = generateRandomArray(n, 0, n); // 生成n个[0,n]的随机数 int *arr2 = copyArray(arr1, n); int *arr3 = copyArray(arr2, n); testSort(arr1, n, "QuickSortOneWays1", QuickSortOneWays1); testSort(arr2, n, "QuickSortTwoWays", QuickSortTwoWays); testSort(arr3, n, "QuickSortThreeWays", QuickSortThreeWays); free(arr1);free(arr2);free(arr3);
// 生成 几乎有序的数组 最多只有2*100个元素无序 int swapTimes = 100; printf("\ntest generateNearlyOrderArray, size = %d, sawpTimes = %d\n",n, swapTimes); int *arr4 = generateNearlyOrderArray(n, swapTimes); int *arr5 = copyArray(arr4, n); int *arr6 = copyArray(arr4, n); testSort(arr4, n, "QuickSortOneWays1", QuickSortOneWays1); testSort(arr5, n, "QuickSortTwoWays", QuickSortTwoWays); testSort(arr6, n, "QuickSortThreeWays", QuickSortThreeWays); free(arr4);free(arr5);free(arr6);
// 生成大量重复元素数组 [0,10] printf("\ntest generateRandomArray, size = %d, random range [0, %d]\n",n, 10); int *arr7 = generateRandomArray(n, 0, 10); int *arr8 = copyArray(arr7, n); int *arr9 = copyArray(arr7, n); testSort(arr7, n, "QuickSortOneWays1", QuickSortOneWays1); testSort(arr8, n, "QuickSortTwoWays", QuickSortTwoWays); testSort(arr9, n, "QuickSortThreeWays", QuickSortThreeWays); free(arr7);free(arr8);free(arr9); return0; }