【排序算法(二)】——冒泡排序、快速排序和归并排序—>深层解析

简介: 【排序算法(二)】——冒泡排序、快速排序和归并排序—>深层解析

前言:

       接上篇,排序算法除了选择排序(希尔排序)和插入排序(堆排序)之外,还用交换排序(冒泡排序、快速排序)和归并排序已经非比较排序,本篇来深层解析这些排序算法

一、交换排序

       1.1、冒泡排序

       冒泡排序,这个再熟悉不过了,学校中老师讲的第一个排序就是冒泡排序;直接看代码

代码如下:

//冒泡排序
void BubbleSort(int* arr, int n)
{
  int exchange = 0;
  for (int i = 0; i < n; i++)
  {
    for (int j = 0; j < n - i - 1; j++)
    {
      if (arr[j] > arr[j + 1])
      {
        exchange = 1;
        Swap(&arr[j], &arr[j + 1]);
      }
    }
    if (exchange == 0)
    {
      break;
    }
  }
}

       时间复杂度:O(n^2);空间复杂度O(1)。

       1.2、快速排序

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

实现快速排序主要框架如下:

//快速排序
void QuickSort(int* a, int left, int right)
{
  if (left >= right) {
    return;
  }
  //_QuickSort⽤于按照基准值将区间[left,right)中的元素进⾏划分
  int meet = _QuickSort(a, left, right);
  QuickSort(a, left, meet - 1);
  QuickSort(a, meet + 1, right);
}

下面的不同方法指的是找基准值的方法不同而已。

1.2.1、hoare版本

思路:

       1.  创建左右指针,却基准值

       2.  从左到右找出比基准值大的数据,从右到左找出比基准值小的数据,左右指针数据交换,进入下一次循环

这里可能有一些问题,

       1.跳出循环后,right位置的值一定不大于key?

       当left > right时,即right走到了left的左侧,而left走过的位置值都不大于key,因此right此时指向的数据一定不大于key

       2.为什么left或者right指定的数据与key值相等也要交换?

       这里如果,数组中大量的数据都相等,不进行交换的话,就无法进行有效的分割数组。

代码实现:

//快速排序
int _QuickSort1(int* arr, int left, int right)
{
  int key = arr[left];
  int mid = left;
  left++;
  while (left <= right)
  {
    //左边找大
    while (left<=right && arr[left] < key)
    {
      left++;
    }
    //右边找小
    while (left <= right && arr[right] > key)
    {
      right--;
    }
    if (left <= right)
    {
      Swap(&arr[left], &arr[right]);
      left++;
      right--;
    }
  }
  Swap(&arr[mid], &arr[right]);
  return right;
}

1.2.2、挖坑法

思路:

       创建左右指针;首先从右向左找出比基准值小的数据,找到后立即放入左边 "坑" 中,当前位置变为新的 "坑",然后从左往右找出比基准值大的数据,找到后立即放入右边坑中,当前位置变为新的 "坑",结束循环后将最开始存储的分界值放入当前的 "坑"中,返回当前"坑"下标。

代码实现如下:

int _QuickSort2(int* arr, int left, int right)
{
  int tmp = arr[left];
  int hole = left;
  left++;
  while (left < right)
  {
    //从右边开始找
    while (left < right && arr[right] > tmp)
    {
      right--;
    }
    arr[hole] = arr[right];
    hole = right;
    while (left < right && arr[left] < tmp)
    {
      left++;
    }
    arr[hole] = arr[left];
    hole = left;
  }
  arr[hole] = tmp;
  return hole;
}

1.2.3、lomuto指针法

思路:

       创建前后指针,从左往右找比基准值小的进行交换,使得小的都排在基准值的左边。

代码实现:

int _QuickSort(int* arr, int left, int right)
{
  int prev = left, pcur = left + 1;
  int key = left;
  while (pcur <= right)
  {
    if (arr[pcur] < arr[key] && ++prev != pcur)
    {
      Swap(&arr[prev], &arr[pcur]);
    }
    pcur++;
  }
  Swap(&arr[key], &arr[prev]);
  return prev;
}

二、归并排序

思路:

       归并排序,是建立在归并操作上的一种有效的排序算法,该算法是采用分治法的一个典型的应用。将已有序的子序列合并,得到完全有序的序列;即先让每一个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路合并。

代码实现:

//归并排序
void _MergeSort(int* arr, int left, int right,int* tmp)
{
  if (left >= right)
  {
    return;
  }
 
  int mid = (left + right) / 2;
  _MergeSort(arr, left, mid, tmp);
  _MergeSort(arr, mid+1, right, tmp);
 
 
  //合并数组
  int begin1 = left, end1 = mid;
  int begin2 = mid + 1, end2 = right;
  int index = left;
  while (begin1 <= end1 && begin2 <= end2)
  {
    if (arr[begin1] < arr[begin2])
    {
      tmp[index++] = arr[begin1++];
    }
    else {
      tmp[index++] = arr[begin2++];
    }
  }
  while (begin1 <= end1)
  {
    tmp[index++] = arr[begin1++];
  }
  while (begin2 <= end2)
  {
    tmp[index++] = arr[begin2++];
  }
  //将tmp数据拷贝回arr中
  for (int i = left; i <= right; i++)
  {
    arr[i] = tmp[i];
  }
}
void MergeSort(int* arr, int n)
{
  int* tmp = (int*)malloc(sizeof(int) * n);
 
  _MergeSort(arr, 0, n - 1, tmp);
 
  free(tmp);
}

三、非比较排序

非比较排序,就是不进行比较数据来进行排序。

       计数排序

1>  统计相同元素出现次数

2>  根据统计的结果将序列回收到原来的序列中

这里又会存在一些问题,比如如果数据是负数,那该怎样开辟空间?

       这里,我们开辟空间大小为数组数据最大值和最小值之差。

然后在将统计结果返回到原来数组当中时,让数组下标加上原数组最小值即可。

//计数排序
void CountSort(int* arr, int n)
{
  int max = arr[0];
  int min = arr[0];
  for (int i = 0; i < n; i++)
  {
    if (arr[i] > max)
    {
      max = arr[i];
    }
    if (arr[i] < min)
    {
      min = arr[i];
    }
  }
 
  //开辟空间
  int range = max - min + 1;
  int* tmp = (int*)calloc(sizeof(int),range);
  if (tmp == NULL)
  {
    perror("calloc fail");
    return;
  }
 
  for (int i = 0; i < n; i++)
  {
    tmp[arr[i] - min]++;
  }
  int j = 0;
  for (int i = 0; i < range; i++)
  {
    while (tmp[i]--)
    {
      arr[j++] = i + min;
    }
  }
}

各种排序算法的算法复杂度和稳定性分析

到这里,排序算法就结束了,希望你能有所收获

感谢各位大佬支持并指出问题,

                       如果本篇内容对你有帮助,可以一键三连支持以下,感谢支持!!!

相关文章
|
9月前
|
存储 人工智能 算法
从零掌握贪心算法Java版:LeetCode 10题实战解析(上)
在算法世界里,有一种思想如同生活中的"见好就收"——每次做出当前看来最优的选择,寄希望于通过局部最优达成全局最优。这种思想就是贪心算法,它以其简洁高效的特点,成为解决最优问题的利器。今天我们就来系统学习贪心算法的核心思想,并通过10道LeetCode经典题目实战演练,带你掌握这种"步步为营"的解题思维。
|
10月前
|
机器学习/深度学习 人工智能 搜索推荐
从零构建短视频推荐系统:双塔算法架构解析与代码实现
短视频推荐看似“读心”,实则依赖双塔推荐系统:用户塔与物品塔分别将行为与内容编码为向量,通过相似度匹配实现精准推送。本文解析其架构原理、技术实现与工程挑战,揭秘抖音等平台如何用AI抓住你的注意力。
2424 7
从零构建短视频推荐系统:双塔算法架构解析与代码实现
|
10月前
|
机器学习/深度学习 存储 算法
动态规划算法深度解析:0-1背包问题
0-1背包问题是经典的组合优化问题,目标是在给定物品重量和价值及背包容量限制下,选取物品使得总价值最大化且每个物品仅能被选一次。该问题通常采用动态规划方法解决,通过构建二维状态表dp[i][j]记录前i个物品在容量j时的最大价值,利用状态转移方程避免重复计算子问题,从而高效求解最优解。
938 1
|
10月前
|
算法 搜索推荐 Java
贪心算法:部分背包问题深度解析
该Java代码基于贪心算法求解分数背包问题,通过按单位价值降序排序,优先装入高价值物品,并支持部分装入。核心包括冒泡排序优化、分阶段装入策略及精度控制,体现贪心选择性质,适用于可分割资源的最优化场景。
499 1
贪心算法:部分背包问题深度解析
|
10月前
|
机器学习/深度学习 边缘计算 人工智能
粒子群算法模型深度解析与实战应用
蒋星熠Jaxonic是一位深耕智能优化算法领域多年的技术探索者,专注于粒子群优化(PSO)算法的研究与应用。他深入剖析了PSO的数学模型、核心公式及实现方法,并通过大量实践验证了其在神经网络优化、工程设计等复杂问题上的卓越性能。本文全面展示了PSO的理论基础、改进策略与前沿发展方向,为读者提供了一份详尽的技术指南。
粒子群算法模型深度解析与实战应用
|
10月前
|
机器学习/深度学习 资源调度 算法
遗传算法模型深度解析与实战应用
摘要 遗传算法(GA)作为一种受生物进化启发的优化算法,在复杂问题求解中展现出独特优势。本文系统介绍了GA的核心理论、实现细节和应用经验。算法通过模拟自然选择机制,利用选择、交叉、变异三大操作在解空间中进行全局搜索。与梯度下降等传统方法相比,GA不依赖目标函数的连续性或可微性,特别适合处理离散优化、多目标优化等复杂问题。文中详细阐述了染色体编码、适应度函数设计、遗传操作实现等关键技术,并提供了Python代码实现示例。实践表明,GA的成功应用关键在于平衡探索与开发,通过精心调参维持种群多样性同时确保收敛效率
机器学习/深度学习 算法 自动驾驶
1461 0
|
10月前
|
机器学习/深度学习 人工智能 资源调度
大语言模型的核心算法——简要解析
大语言模型的核心算法基于Transformer架构,以自注意力机制为核心,通过Q、K、V矩阵动态捕捉序列内部关系。多头注意力增强模型表达能力,位置编码(如RoPE)解决顺序信息问题。Flash Attention优化计算效率,GQA平衡性能与资源消耗。训练上,DPO替代RLHF提升效率,MoE架构实现参数扩展,Constitutional AI实现自监督对齐。整体技术推动模型在长序列、低资源下的性能突破。
1116 8
|
10月前
|
算法 API 数据安全/隐私保护
深度解析京东图片搜索API:从图像识别到商品匹配的算法实践
京东图片搜索API基于图像识别技术,支持通过上传图片或图片URL搜索相似商品,提供智能匹配、结果筛选、分页查询等功能。适用于比价、竞品分析、推荐系统等场景。支持Python等开发语言,提供详细请求示例与文档。
|
12月前
|
机器学习/深度学习 人工智能 编解码
AI视觉新突破:多角度理解3D世界的算法原理全解析
多视角条件扩散算法通过多张图片输入生成高质量3D模型,克服了单图建模背面细节缺失的问题。该技术模拟人类多角度观察方式,结合跨视图注意力机制与一致性损失优化,大幅提升几何精度与纹理保真度,成为AI 3D生成的重要突破。
1462 0

热门文章

最新文章

推荐镜像

更多
  • DNS