快排思想

快排类似铲土,把大于基准的铲到右边,小于基准的铲到左边,一次快排结束,会得到基准的正确位置
其他元素的位置正不正确未知,接下来把基准两侧的数组,分别递归调用快排。直至所有元素归位

循环过程中**

  1. 从 j 开始向左搜索,搜索比基准小的数字,**铲到 i 所在位置**
  2. j 铲完,i开始向右搜索,搜索比基准大的数字,**铲到 j 所在的位置**
  3. i,j 轮流互相铲“ 土元素 “,直至 i = j,**此位置放置基准**
  4. 一次循环结束

时间复杂度分析


  1. 最糟糕情况下时间复杂度是O(n²)

解释:基准取第一项时,是最糟的情况,完成每层需要的时间是 n,栈的高度是 n,时间复杂度就是n²

  1. - **备注:**第一项指的是基准取了**最小值 **[ **升序** ] 或者**最大值 **[ **降序** ]
  1. 平均的复杂度是O(nlogn)

解释:取中间值时候,完成每层的时间是n,调用栈的高度变成logn,这个时候时间复杂度是nlogn

  1. - **备注:**这里的 n logn , 分别代表了**调用栈的高度**和**完成每层的时间**

快排铲土过程演示

  1. **起始数组** ![image.png](https://cdn.nlark.com/yuque/0/2020/png/2955945/1608650012646-e4f459cb-b8b1-4692-81db-7d15fe61375c.png#align=left&display=inline&height=65&margin=%5Bobject%20Object%5D&name=image.png&originHeight=130&originWidth=772&size=17350&status=done&style=none&width=386)

j 从右边向左搜索,比基准小的数字,铲到 i 的位置

基准** -1** image.png

i = j 搜索结束,将基准放入 i = j 位置
image.png

j 从右边向左搜索,比基准小的数字,铲到 i 的位置
image.png**
2FT6$_20``(3%7]KC68F72I.png
j 铲到 i 的位置
image.png
j 铲完,i 向右搜索,搜索比基准大的数字,铲到 j 处
image.png

基准12位置正确**,因此将数组分成两部分,分别递归快排求解**

快排代码

铲土流快排代码

  1. #include<iostream>
  2. #pragma warning(disable:4996)
  3. using namespace std;
  4. //快速排序算法(从小到大)
  5. //arr:需要排序的数组,begin:需要排序的区间左边界,end:需要排序的区间的右边界 [begin,end]
  6. int partition(int arr[], int begin, int end) {
  7. int temp = arr[begin]; //将区间的第一个数作为基准数
  8. int i = begin; //从左到右进行查找时的“指针”,指示当前左位置
  9. int j = end; //从右到左进行查找时的“指针”,指示当前右位置
  10. //不重复遍历
  11. while (i < j)
  12. {
  13. //当右边的数大于基准数时,略过,继续向左查找
  14. //不满足条件时跳出循环,此时的j对应的元素是小于基准元素的
  15. while (i<j && arr[j] > temp)
  16. j--;
  17. //将右边小于等于基准元素的数填入右边相应位置
  18. arr[i] = arr[j];
  19. //当左边的数小于等于基准数时,略过,继续向右查找
  20. //(重复的基准元素集合到左区间)
  21. //不满足条件时跳出循环,此时的i对应的元素是大于等于基准元素的
  22. while (i < j && arr[i] <= temp)
  23. i++;
  24. //将左边大于基准元素的数填入左边相应位置
  25. arr[j] = arr[i];
  26. }
  27. //将基准元素填入相应位置
  28. arr[i] = temp;
  29. //此时的i即为基准元素的位置
  30. return i;
  31. }
  32. void quickSort(int arr[], int begin, int end)
  33. {
  34. //如果区间不只一个数
  35. if (begin < end)
  36. { //获取基准的位置
  37. int i = partition(arr, begin, end);
  38. //对基准元素的左边子区间进行相似的快速排序
  39. quickSort(arr, begin, i - 1);
  40. //对基准元素的右边子区间进行相似的快速排序
  41. quickSort(arr, i + 1, end);
  42. }
  43. //如果区间只有一个数或者end>begein,则返回
  44. else
  45. return;
  46. }
  47. int main()
  48. {
  49. int num[12] = { 23,45,17,11,13,89,72,26,3,17,11,13 };
  50. int n = sizeof(num) / sizeof(num[0]);//计算数组长度
  51. quickSort(num, 0, n - 1);
  52. cout << "排序后的数组为:" << endl;
  53. for (int i = 0; i < n; i++)
  54. cout << num[i] << ' ';
  55. cout << endl;
  56. system("pause");
  57. return 0;
  58. }

🚩这种 Split 方式比较有趣,一个 for 循环到底,因此也放这里了
[ 解释 ] : 我的 Split** 就是互相来回铲土

下面解释这 Split 独特之处

  1. 将首元素 nums[0],当做基准
  2. j 指针 下标1开始移动至数组末尾,只处理比基准小的元素
    • j 下标元素比基准小,则和 i++,swap( a[i] , a[j] )
  3. 循环交换 swap( a[low], a[i] )

🍖例子流程图
image.png
image.png

  1. #include<stdio.h>
  2. void swap(int &a, int &b)
  3. {
  4. int t = a;
  5. a = b;
  6. b = t;
  7. }
  8. //划分数组的函数
  9. int split(int a[], int low, int high)
  10. {
  11. int i = low; //i指向比较元素的期望位置
  12. int x = a[i]; //将该数组第一个元素设置为比较元素
  13. //从数组的第二个元素起开始遍历,若找到的元素大于比较元素,则跳过
  14. for(int j = low+1;j<=high;j++)
  15. //若找到了小于比较元素的数,则将其与前面较大的数进行交换
  16. if (a[j] <= x)
  17. {
  18. i++;
  19. swap(a[i], a[j]);
  20. }
  21. swap(a[low], a[i]); //将比较元素交换到期望位置
  22. return i;
  23. }
  24. //快速排序
  25. void quicksort(int a[], int low, int high)
  26. {
  27. if (low < high)
  28. {
  29. int i = split(a, low, high); //划分数组并获得比较元素位置
  30. quicksort(a, low, i - 1); //对比较元素左边进行排序
  31. quicksort(a, i + 1, high); //对比较元素右边进行排序
  32. }
  33. }
  34. int main()
  35. {
  36. int a[] = { 5,7,1,6,4,8,3,2 };
  37. int length = sizeof(a) / sizeof(a[0]);
  38. quicksort(a, 0, length - 1);
  39. for (int i = 0; i < length; i++)
  40. printf("%d ", a[i]);
  41. printf("\n");
  42. return 0;
  43. }