数据结构各内部排序算法总结对比及动图演示(插入排序、冒泡和快速排序、选择排序、堆排序、归并排序和基数排序等)1

简介: 数据结构各内部排序算法总结对比及动图演示(插入排序、冒泡和快速排序、选择排序、堆排序、归并排序和基数排序等)1


一、插入排序

插入排序包括直接插入排序,折半插入排序、希尔排序直接插入排序就是简单粗暴的插入,折半排序是利用了二分查找的插入排序,希尔排序是先局部后整体的插入排序。

其算法的主要思想就是每次将一个待排序的记录按其关键字大小插入到前面已经排好序的子序列,直到全部记录插入完成。

1.直接插入排序

①算法的执行过程:

对于待排序表L[1...n]假设在某个状态下,待排序元素为L(i),则L[1...i-1]为已经排好序的序列,L[i+1...n]为无序序列。

L(i)依次与L(i-1)...L(1)相比较,找出L(i)在有序序列中要插入的位置k

L[k...i-1]中的所有元素依次后移一个位置

L(i)复制到L(k)i后移,重复2.直到有序序列长度为n

②算法执行过程可视化演示:


③算法代码:

void InsertSort(ElemType A[], int n){
  for(int i = 2; i <= n; i++){        //默认首个元素为有序序列,将2~n位置的关键字依次插入 
    if(A[i] < A[i-1]){            //如果待插入元素小于有序序列最大元素,则需要插入 
      A[0] = A[i];            //A[0]为“哨兵”,用来存放待插入元素 
      for(int j = i-1; A[0] < A[j]; j--)  //从后往前查找待插入的位置 
        A[j+1] = A[j];          //依次向后移动 
      A[j+1] = A[0];            //找到位置之后,将待排序元素插入有序序列 
    }
  }
}

④性能分析:

1.空间效率:使用常数个辅助单元,空间复杂度为O(1)

2.时间效率:进行了n-1趟插入操作,每趟操作都分为比较和移动元素,所以与表的初始状态有关
最好情况下:表已经有序只需比较不需移动元素,此时时间复杂度为O(n);

最坏情况下:此时为逆序,比较次数为2+3+……+n,总的移动次数为(2+1)+(3+1)+……+(n+1)

平均情况下:出现概率随机取最好和最坏的平均值,总的比较和移动次数约为(1/4)n2.所以插入排序算法的时间复杂度为O(n2)

3.稳定性: 稳定

4.适用性:适用于线性表为顺序存储或链式存储的情况。

2.折半插入排序

①算法的执行过程: 总体过程与上一个类似,只是在寻找插入位置的时候使用的二分查找算法

②算法执行过程可视化演示: 与上一个相同。

③算法代码:

void BinaryInsertSort(ElemType A[], int n){
  int low, high, mid;
  for(int i = 2; i <= n; i++){      //默认首个元素为有序序列,将2~n位置的关键字依次插入
    A[0] = A[i];            //A[0]为“哨兵”,用来存放待插入元素
    low = 1, high = i-1;
    while(low <= high){         //low=high时表明查找到要插入的位置 
      mid = (low+high)/2;
      //为了保证稳定,相等的情况需要查找右半子表 
      if(A[mid] > A[0]) high = mid-1; //查找左半子表 
      else low = mid+1;       //查找右半子表 
    }
    for(int j = i-1; j >= high+1; j--)  //从后往前查找待插入的位置
      A[j+1] = A[j];          //依次向后移动
    A[high+1] = A[0];         //将待排序元素插入有序序列
  }
}

④性能分析:

1.空间效率:与直接插入排序相同空间复杂度为O(1)

2.时间效率:进行了n-1趟插入操作,每趟操作都分为比较和移动元素,比较操作与表的初始状态无关,为O(n log2n),移动次数取决于初始状态,所以折半插入排序时间复杂度为O(n2)
3.稳定性: 不稳定

4.适用性:仅适用于线性表为顺序存储的情况。

3.希尔排序

希尔排序的由来:当待排序序列为基本有序时,插入排序复杂度可以提高至O(n),所以我们可以让整体基本有序,也就是说部分有序,最后使用插入排序进行排序。由此得出希尔排序,也称缩小增量排序
①算法的执行过程:

希尔排序思想即先局部后整体,首先设置增量d,把待排序的表分为k个子表对每个子表排序后,增量d变为原来的一半直到减小为d=1.

此时的表已经基本有序,进行最后一趟排序之后得到有序序列,这里在每个子表中使用的排序算法仍是插入排序.

②算法执行过程演示:


③算法代码:

void ShellSort(ElemType A[], int n){
  //通过增量d把序列分为多个子表,外层for循环控制增量的变化 
  for(int dk = n/2; dk >= 1; dk /= 2){
    //对于每一个增量得到的子表进行插入排序 
     for(int i = dk+1; i <= n; i++){
      if(A[i] < A[i-dk]){     //如果待插入元素小于有序序列最大元素,则需要插入       
        A[0] = A[i];      //将元素暂存到A[0],但在这里并没有起到哨兵的作用 
        for(int j = i-dk; j > 0 && A[0] < A[j]; j -= dk)
          A[j+dk] = A[j];   //依次向后移动
        A[j+dk] = A[0];     //将待排序元素插入有序序列
      }
    }
  }
} 

④性能分析:

1.空间效率:使用常数个辅助单元,空间复杂度为O(1)

2.时间效率:由于希尔排序的复杂度依赖增量序列的函数,所以时间复杂度分析比较困难。当n在某个特定范围时,其时间复杂度为O(n1.3),最坏情况下为O(n2)。
3.稳定性:不稳定

4.适用性:仅适用于线性表为顺序存储的情况。


二、交换排序

此类排序是根据序列中两个元素关键字的比较结果来对换这两个记录在序列中的位置。

1.冒泡排序

①算法的执行过程:

冒泡排序是一种基于比较的简单交换排序,会进行多轮交换,在每趟排序中,从后往前两两比较相邻元素的值,若为逆序,则交换他们。
在每趟排序完成后,都会将当前待排序序列中最小的元素放到第一个位置(或最大的元素放到最后一个位置)。

最多进行n-1趟处理后,所有元素就能排好。第i趟排序要进行n-i次比较。

②算法执行过程可视化演示:


③算法代码:

void BubbleSort(ElemType A[], int n){
  for(int i = 0; i < n-1; i++){   //一共n-1趟
    for(int j = 0; j < n-1-i; j++){ //对应每一趟的比较
      if(A[j] > A[j+1]){      //若为逆序则交换
        swap(A[j], A[j+1]);
      }
    }
  } 
}

④性能分析:

1.空间效率:使用常数个辅助单元,空间复杂度为O(1)

2.时间效率:最好情况下时间复杂度为O(n),最坏情况下初始为逆序,需要n-1趟排序,第i趟需要n-i次关键字的比较,每次比较都需要移动元素3次来交换元素位置,所以最坏情况和平均情况复杂度都为O(n2).
3.稳定性:稳定

4.适用性:适用于线性表为顺序存储和链式存储的情况。

拓展(链式存储的冒泡排序):

void sort_list(PNODE pHead){
  int i,j,t;
  PNODE p, q;
  int len= length_list(pHead);
  for (i=0,p=pHead->pNext;i<len-1;++i,p=p->pNext){
    for (j=i+1,q=p->pNext;j<len;++j,q=q->pNext){
      if (p->data > q->data){
        t = p->data;
        p->data = q->data;
        q->data = t;
      }
    }
  }
}

2.快速排序

①算法的执行过程:

  • 快速排序是一种基于分治思想的交换排序
  • 在待排表L[1...n]中任取一个元素pivot作为枢轴(或基准)
    通过一趟排序将待排序表划分为两个部分L[1...k-1]L[k+1...n],使得L[1...k-1]中所有元素小于pivot,L[k+1...n]中所有元素大于pivot。
  • 此时的枢轴元素privot已经放到了最终的位置上。
  • 递归的对两个子表重复以上步骤,直到每个子表只有一个元素或为空。

②算法执行过程演示:


③算法代码:

//划分操作 
int Partition(ElemType A[], int low, int high){
  ElemType pivot = A[low];  //定义第一个元素为枢轴元素 
  while(low < high){
    while(low < high && A[high] > pivot) high--;
    A[low] = A[high];   //从后往前找到第一个小于枢轴的元素放到左边 
    while(low < high && a[low] < pivot) low++;
    A[high] = A[low];   //从前往后找到第一个大于枢轴的元素放到右边 
  }
  A[low] = pivot;       //将枢轴元素放到最终的位置 
  return low;         //返回枢轴元素的位置 
}
//递归的进行快排
void QuickSort(ElemType A[], int low, int high){
  if(low < high){             //如果两个指针的位置相反或者相等,表明递归结束 
    int pos = Partition(A, low, high);  //找到枢轴元素的位置 
    QuickSort(A, low, pos-1);     //递归排序左子表 
    QuickSort(A, pos+1, high);      //递归排序右子表 
  }
} 

④性能分析:

1.空间效率:由于快排是递归的,所以空间复杂度与递归深度有关。
二分递归空间复杂度最好情况下为O(log2n),最坏情况下进行n-1次调用,深度为O(n)
2.时间效率:与划分的好坏有关

最坏情况下每层递归都是最大限度的不对称(两个区域分别包含n-1和0个元素),复杂度为O(n2)也就是对应初始排序表基本有序或基本逆序

最理想情况每层递归能能完美的划分,复杂度为O(nlog2n)。得到的两个子问题规模都小于n/2

3.稳定性:不稳定

4.适用性:适用于线性表为顺序存储的情况。

相关文章
|
2月前
|
搜索推荐 C语言
【排序算法】快速排序升级版--三路快排详解 + 实现(c语言)
本文介绍了快速排序的升级版——三路快排。传统快速排序在处理大量相同元素时效率较低,而三路快排通过将数组分为三部分(小于、等于、大于基准值)来优化这一问题。文章详细讲解了三路快排的实现步骤,并提供了完整的代码示例。
62 4
|
2月前
|
存储 搜索推荐 Python
用 Python 实现快速排序算法。
快速排序的平均时间复杂度为$O(nlogn)$,空间复杂度为$O(logn)$。它在大多数情况下表现良好,但在某些特殊情况下可能会退化为最坏情况,时间复杂度为$O(n^2)$。你可以根据实际需求对代码进行调整和修改,或者尝试使用其他优化策略来提高快速排序的性能
128 61
|
3月前
|
算法 搜索推荐 Shell
数据结构与算法学习十二:希尔排序、快速排序(递归、好理解)、归并排序(递归、难理解)
这篇文章介绍了希尔排序、快速排序和归并排序三种排序算法的基本概念、实现思路、代码实现及其测试结果。
51 1
|
3月前
|
数据可视化 搜索推荐 Python
Leecode 刷题笔记之可视化六大排序算法:冒泡、快速、归并、插入、选择、桶排序
这篇文章是关于LeetCode刷题笔记,主要介绍了六大排序算法(冒泡、快速、归并、插入、选择、桶排序)的Python实现及其可视化过程。
25 0
|
12天前
|
机器学习/深度学习 算法
基于改进遗传优化的BP神经网络金融序列预测算法matlab仿真
本项目基于改进遗传优化的BP神经网络进行金融序列预测,使用MATLAB2022A实现。通过对比BP神经网络、遗传优化BP神经网络及改进遗传优化BP神经网络,展示了三者的误差和预测曲线差异。核心程序结合遗传算法(GA)与BP神经网络,利用GA优化BP网络的初始权重和阈值,提高预测精度。GA通过选择、交叉、变异操作迭代优化,防止局部收敛,增强模型对金融市场复杂性和不确定性的适应能力。
145 80
|
5天前
|
机器学习/深度学习 算法
基于遗传优化的双BP神经网络金融序列预测算法matlab仿真
本项目基于遗传优化的双BP神经网络实现金融序列预测,使用MATLAB2022A进行仿真。算法通过两个初始学习率不同的BP神经网络(e1, e2)协同工作,结合遗传算法优化,提高预测精度。实验展示了三个算法的误差对比结果,验证了该方法的有效性。
|
8天前
|
机器学习/深度学习 数据采集 算法
基于PSO粒子群优化的CNN-GRU-SAM网络时间序列回归预测算法matlab仿真
本项目展示了基于PSO优化的CNN-GRU-SAM网络在时间序列预测中的应用。算法通过卷积层、GRU层、自注意力机制层提取特征,结合粒子群优化提升预测准确性。完整程序运行效果无水印,提供Matlab2022a版本代码,含详细中文注释和操作视频。适用于金融市场、气象预报等领域,有效处理非线性数据,提高预测稳定性和效率。
|
4天前
|
算法
基于梯度流的扩散映射卡尔曼滤波算法的信号预处理matlab仿真
本项目基于梯度流的扩散映射卡尔曼滤波算法(GFDMKF),用于信号预处理的MATLAB仿真。通过设置不同噪声大小,测试滤波效果。核心代码实现数据加载、含噪信号生成、扩散映射构建及DMK滤波器应用,并展示含噪与无噪信号及滤波结果的对比图。GFDMKF结合非线性流形学习与经典卡尔曼滤波,提高对非线性高维信号的滤波和跟踪性能。 **主要步骤:** 1. 加载数据并生成含噪测量值。 2. 使用扩散映射捕捉低维流形结构。 3. 应用DMK滤波器进行状态估计。 4. 绘制不同SNR下的轨迹示例。
|
9天前
|
机器学习/深度学习 算法 索引
单目标问题的烟花优化算法求解matlab仿真,对比PSO和GA
本项目使用FW烟花优化算法求解单目标问题,并在MATLAB2022A中实现仿真,对比PSO和GA的性能。核心代码展示了适应度计算、火花生成及位置约束等关键步骤。最终通过收敛曲线对比三种算法的优化效果。烟花优化算法模拟烟花爆炸过程,探索搜索空间,寻找全局最优解,适用于复杂非线性问题。PSO和GA则分别适合快速收敛和大解空间的问题。参数调整和算法特性分析显示了各自的优势与局限。
|
3天前
|
算法 5G
基于MSWA相继加权平均的交通流量分配算法matlab仿真
本项目基于MSWA(Modified Successive Weighted Averaging)相继加权平均算法,对包含6个节点、11个路段和9个OD对的交通网络进行流量分配仿真。通过MATLAB2022A实现,核心代码展示了迭代过程及路径收敛曲线。MSWA算法在经典的SUE模型基础上改进,引入动态权重策略,提高分配结果的稳定性和收敛效率。该项目旨在预测和分析城市路网中的交通流量分布,达到用户均衡状态,确保没有出行者能通过改变路径减少个人旅行成本。仿真结果显示了27条无折返有效路径的流量分配情况。