数据结构第六课 -----排序-2

简介: 数据结构第六课 -----排序

数据结构第六课 -----排序-1

https://developer.aliyun.com/article/1498935


快速排序

hoare版本

快速排序是Hoare于1962年提出的一种二叉树结构的交换排序方法,其基本思想为:任取待排序元素序列中的某元素作为基准值,按照该排序码将待排序集合分割成两子序列,左子序列中所有元素均小于基准值,右子序列中所有元素均大于基准值,然后最左右子序列重复该过程,直到所有元素都排列在相应位置上为止

f5b767a426b56e2489c1e70594d17aa3_8d3a2ed9dd634e0bb0f0c82707818405.png


这个图可能有点简陋

时间复杂度:每一次都会把当前数组的每个元素遍历一遍,然后再把key交换, 需要进行log(n)次递归

时间复杂度是:O(n*log(n))


复杂的话,就如同这个一样,这种情况就是有n层, 时间复杂度就是 1+2+3+…+n, 所以时间复杂度就是O(n^2)

//快速排序
void QuickSrot(int* a, int begin, int end)
{
  //当只有一个元素就不用进行了
  if (begin >= end)
    return;
  int key = begin;
  int left = begin;//这里不能begin加1 否则在遇到有序的时候就会排序出错
  int right = end;
  while (left < right)
  {
    // 找最小
    while (left < right)
    {
      if (a[right] < a[key])
      {
        break;
      }
      right--;
    }

    // 找最大
    while (left < right)
    {
      if (a[left] > a[key])
      {
        break;
      }
      left++;
    }
    
    excheng(&a[right], &a[left]);
  }
  excheng(&a[right], &a[key]);
  //左
  QuickSrot(a, begin, right - 1);
  // 右
  QuickSrot(a, right + 1, end);
}


优化点

三数取中

思路:

我们可以在数组的前后和中间选取中位数,然后把中位数和开头进行交换,

int TriNum(int *a,int begin, int end)
{
  int mid = (begin - end) / 2 + end;
  if (begin > end)
  {
    if (end > mid)
    {
      return end;
    }
    else if(begin < mid)
    {
      return begin;
    }
    return mid;
  }
  else
  {
    if (begin > mid)
    {
      return begin;
    }
    else if (end < mid)
    {
      return end;
    }
    else
      return mid;
  }
}
//快速排序
void QuickSrot(int* a, int begin, int end)
{
  //当只有一个元素就不用进行了
  if (begin >= end)
    return;
  //三数取中
  int key = 0;
  key = TriNum(a, begin, end);
  exchange(&a[begin], &a[key]);
  key = begin;
  int left = begin;
  int right = end;
  //普通方法
  //int key = begin;
  //int left = begin;//这里不能begin加1 否则在遇到有序的时候就会排序出错
  //int right = end;
  while (left < right)
  {
    // 找最小
    while (left < right)
    {
      if (a[right] < a[key])
      {
        break;
      }
      right--;
    }

    // 找最大
    while (left < right)
    {
      if (a[left] > a[key])
      {
        break;
      }
      left++;
    }
    
    excheng(&a[right], &a[left]);
  }

  excheng(&a[right], &a[key]);
  //左
  QuickSrot(a, begin, right - 1);
  // 右
  QuickSrot(a, right + 1, end);
}


小区间优化

当我们在使用快速排序的时候,一直排序知道递归到还剩下该数组的10%的数没有排序,我们如果使用递归就很对栈的空间浪费很大。那我们可以选择使用插入排序,

//快速排序
void QuickSrot(int* a, int begin, int end)
{
  //当只有一个元素就不用进行了
  if (begin >= end)
    return;
  if (end - begin  + 1 <= 10)
  {
    //插入排序
    InsertSort(a + begin, end - begin + 1);//我们要清楚要从哪里开始插入排序
  }
  else
  {
    //三数取中
    int key = 0;
    key = TriNum(a, begin, end);
    excheng(&a[begin], &a[key]);
    key = begin;
    int left = begin;
    int right = end;
    //普通方法,有可能会栈溢出
    //int key = begin;
    //int left = begin;//这里不能begin加1 否则在遇到有序的时候就会排序出错
    //int right = end;
    while (left < right)
    {
      // 找最小
      while (left < right)
      {
        if (a[right] < a[key])
        {
          break;
        }
        right--;
      }

      // 找最大
      while (left < right)
      {
        if (a[left] > a[key])
        {
          break;
        }
        left++;
      }

      excheng(&a[right], &a[left]);
    }

    excheng(&a[right], &a[key]);
    //左
    QuickSrot(a, begin, right - 1);
    // 右
    QuickSrot(a, right + 1, end);
  }
  
}

挖坑法

//挖坑法
void QuickSrot2(int* a, int begin, int end)
{
  if (begin >= end)
    return;
  if (end - begin + 1 <= 10)
  {
    InsertSort(a + begin, end - begin + 1);
  }
  else
  {
    //三数取中
    int key = TriNum(a, begin, end);
    excheng(&a[key], &a[begin]);
    //坑
    key = begin;
    int num = a[key];
    int left = begin;
    int right = end;
    while (left < right)
    {
      //找小
      while (left < right)
      {
        if (a[right] < num)
        {
          a[key] = a[right];
          key = right;
          break;
        }
        right--;
      }
      //找大
      while (left < right)
      {
        if (a[left] > num)
        {
          a[key] = a[left];
          key = left;
          break;
        }
        left++;
      }
    }
    a[key] = num;
    //左
    QuickSrot(a, begin, right - 1);
    // 右
    QuickSrot(a, right + 1, end);

  }
  
  

}

前后指针版本

思路:

cur遇见比key大的值,cur++

cur遇见比key小的值,prev++,交换prev和cur的值交换,然后cur++

//前后指针版本
// 快速排序版本3
void QuickSrot3(int* a, int begin, int end)
{
  if (begin >= end)
    return;
  int key = TriNum(a, begin, end);
  excheng(&a[key], &a[begin]);
  key = begin;
  int prev = begin;
  int cur = begin + 1;
  while (cur <= end)
  {
    //cur 比较
    if (a[cur] < a[key] && ++prev != cur)//增加++prev != cur可以有效解决相同位置进行交换
    {
      exchange(&a[cur], &a[prev]);
    }
    cur++;
  }
  exchange(&a[key], &a[prev]);
  //左
  QuickSrot(a, begin, prev - 1);
  // 右
  QuickSrot(a, prev + 1, end);
}

疑惑

  1. 为什么相遇位置比key小

原因:是right先走

两种情况:

(1).R遇见L —>(L和R交换后,R先走)R没有找到比key小的,一直走,直到R遇见L,(特殊情况除外)

(2)L遇见R----->(R找到小了),然后L没有找到比key大的,一直走,直到L遇见R,(特殊情况除外)

相关文章
|
5天前
|
存储 人工智能 算法
【C++数据结构——内排序】二路归并排序(头歌实践教学平台习题)【合集】
本关任务是实现二路归并算法,即将两个有序数组合并为一个有序数组。主要内容包括: - **任务描述**:实现二路归并算法。 - **相关知识**: - 二路归并算法的基本概念。 - 算法步骤:通过比较两个有序数组的元素,依次将较小的元素放入新数组中。 - 代码示例(以 C++ 为例)。 - 时间复杂度为 O(m+n),空间复杂度为 O(m+n)。 - **测试说明**:平台会对你编写的代码进行测试,提供输入和输出示例。 - **通关代码**:提供了完整的 C++ 实现代码。 - **测试结果**:展示代码运行后的排序结果。 开始你的任务吧,祝你成功!
25 10
|
5天前
|
搜索推荐 算法 数据处理
【C++数据结构——内排序】希尔排序(头歌实践教学平台习题)【合集】
本文介绍了希尔排序算法的实现及相关知识。主要内容包括: - **任务描述**:实现希尔排序算法。 - **相关知识**: - 排序算法基础概念,如稳定性。 - 插入排序的基本思想和步骤。 - 间隔序列(增量序列)的概念及其在希尔排序中的应用。 - 算法的时间复杂度和空间复杂度分析。 - 代码实现技巧,如循环嵌套和索引计算。 - **测试说明**:提供了测试输入和输出示例,帮助验证代码正确性。 - **我的通关代码**:给出了完整的C++代码实现。 - **测试结果**:展示了代码运行的测试结果。 通过这些内容,读者可以全面了解希尔排序的原理和实现方法。
34 10
|
5天前
|
搜索推荐 C++
【C++数据结构——内排序】快速排序(头歌实践教学平台习题)【合集】
快速排序是一种高效的排序算法,基于分治策略。它的主要思想是通过选择一个基准元素(pivot),将数组划分成两部分。一部分的元素都小于等于基准元素,另一部分的元素都大于等于基准元素。然后对这两部分分别进行排序,最终使整个数组有序。(第一行是元素个数,第二行是待排序的原始关键字数据。本关任务:实现快速排序算法。开始你的任务吧,祝你成功!
25 7
|
3月前
|
算法 搜索推荐 Java
数据结构与算法学习十三:基数排序,以空间换时间的稳定式排序,速度很快。
基数排序是一种稳定的排序算法,通过将数字按位数切割并分配到不同的桶中,以空间换时间的方式实现快速排序,但占用内存较大,不适合含有负数的数组。
48 0
数据结构与算法学习十三:基数排序,以空间换时间的稳定式排序,速度很快。
|
3月前
|
存储 搜索推荐 算法
【用Java学习数据结构系列】七大排序要悄咪咪的学(直接插入,希尔,归并,选择,堆排,冒泡,快排)以及计数排序(非比较排序)
【用Java学习数据结构系列】七大排序要悄咪咪的学(直接插入,希尔,归并,选择,堆排,冒泡,快排)以及计数排序(非比较排序)
37 1
|
3月前
|
搜索推荐 索引
【初阶数据结构】深度解析七大常见排序|掌握底层逻辑与原理(二)
【初阶数据结构】深度解析七大常见排序|掌握底层逻辑与原理
|
3月前
|
搜索推荐 C++
【初阶数据结构】深度解析七大常见排序|掌握底层逻辑与原理(一)
【初阶数据结构】深度解析七大常见排序|掌握底层逻辑与原理
|
3月前
|
算法
蓝桥杯宝藏排序 | 数据结构 | 快速排序 归并排序
蓝桥杯宝藏排序 | 数据结构 | 快速排序 归并排序
05_用一个栈实现另一个栈的排序
05_用一个栈实现另一个栈的排序
|
3月前
|
人工智能 搜索推荐 算法
【初阶数据结构】深度解析七大常见排序|掌握底层逻辑与原理(三)
【初阶数据结构】深度解析七大常见排序|掌握底层逻辑与原理