【数据结构】插入排序

简介: 【数据结构】插入排序

前言

排序皆以升序为例进行讲解



直接插入排序(插排)

生活样例 —— 我们玩扑克牌就是运用了插入排序的思想



一、图例演示



☆二、核心算法思路

从第2个开始,前面只有1个,只有1个就是有序。

把待排序的记录按其关键码值的大小逐个插入到一个 已经排好序的有序序列 中,直到所有的记录插入完为止,得到一个新的有序序列 。



三、算法思路步骤

升序降序同理

  • 单趟
  1. end=0,指向第1个位置(1个也是有序),从第一个开始,将前面的数字都排成有序已确保前方都是 已经排好序的有序序列
  2. end指向的是 序列中已经排好序 的部分的末端
  3. tmp先存储好 a[end+1] ( a[end+1]是要进行插入到前方有序序列的数据 )避免与前面的数据进行比较后 后移,该节点的数据被覆盖
  4. end其后一个数据end+1 向前比较,若 后小于前,则将前面的后移覆盖【也正是因为是 从前往后排 => 前面的数据是已经排好序的有序 => 比它大的都在后面 都往后移】同时end --,进行向前比较遍历。
  5. 遇到比它小的就停下,在a[end+1]的位置 将tmp赋值给a[end+1](因为在整体中是在比较判断外面进行的 end–, 所以不管是否进行后移,都会end–,所以要+回1)
  • 整体
    就用 for(i=0; i<n-1(end到倒数第二个,end+1就是最后一个要进行插排的数了) ;i++) 循环,i 记录控制 end每轮单趟插排所在的 序列中已经排好序 的部分的末端

【注意好这里有 i<n-1 的意义!!控制下标边界问题!!】



四、码源解析

  • 插入排序 —— 单趟
//单趟
void InsertSort(DataType* a, int n) {
  int end = n - 1;
  DataType tmp = a[end];
  while (end > 0) {
    if (tmp < a[end - 1]) {
      a[end] = a[end - 1];
      end--;        //是出循环的条件,别忘了
    }
    else
      break;        //有tmp比所有值都小的情况,则进入了if(tmp < a[end - 1])的条件没有办法出来
  }
  a[end] = tmp;       //将这个单独拎出来
}
  • 插入排序 ——整体
//插入排序 —— 整体  //插入排序:是从前往后,一个一个往前对比遍历排
void InsertSort(DataType* a, int n) {
  for (int i = 0; i < n-1; i++) {
    int end = i;
  DataType tmp = a[end + 1];       //先将要移的数先保存起来,前面发生挪动会将其覆盖
  while (end >= 0) {   //错误示范while (end > 0),那么第一个end=i=0时,进不去while循环 所以要 >=
    if (a[end] > tmp) {
      a[end+1] = a[end];        //比tmp大的都向后挪一位
    }
    else {         //遇到比tmp要小的就停下来,插入tmp
      break;      //有tmp比所有值都小的情况,则进入了if(a[end] > tmp)的条件 没有办法走到else里将tmp赋值给a[end+1],所以把a[end+1] = tmp;放在else里 会在tmp比所有值都小的情况下失效
    }
    --end;        //这里--了后面要+1 (--就到了空出的位置的前一个)
  }   
  a[end+1] = tmp;       //将这个单独拎出来
    }
}

注意点:

  1. a[end+1] = tmp;单独拎出来,为了避免 tmp比所有值都小的情况,则进入了if(a[end] > tmp)的条件 没有办法走到else里将tmp赋值给a[end+1],所以把a[end+1] = tmp;放在else里 会在tmp比所有值都小的情况下失效
  2. 该算法的核心 操纵的是下标,而不是节点本身
  3. 下标边界问题:end 指向有序序列部分的末端,则end+1则是要进行向前比较插入的待排序的数据 ,数组是[0,n-1], end遍历到n-2就已经是最大了,则end+1 则是要进行向前比较插入的待排序的最后一个数据 为n-1。



五、效率分析

1. 时间复杂度

最坏:O(N^2)降序逆序

最好:O(N) 顺序有序

2. 空间复杂度

算法在执行过程中,只有 「移动」变量 时候,需要事先将变量存入临时变量x,而其它没有采用任何的额外空间,所以空间复杂度为 O ( 1 )


与冒泡排序进行对比(同样是小数据排序):冒泡 < 插排

排前面都有序,就最后两个数字顺序颠倒

进行比较的次数

  • 冒泡:n-1 [进行第一次遍历] + n-2 [第一次遍历排好后,再进行第二次遍历检查一遍,排好了结束]
  • 插排:n-1 [从头开始遍历比较一遍] + 1 [多插一次]


得出结论:

时间复杂度上 O(N)一样

空间复杂度上O(1)也一样

但在细节上,局部有序,有一小段有序,插排的情况都会好很多,中间挪动次数也会少很多


目录
相关文章
|
3月前
|
算法 搜索推荐
数据结构与算法学习十一:冒泡排序、选择排序、插入排序
本文介绍了冒泡排序、选择排序和插入排序三种基础排序算法的原理、实现代码和测试结果。
28 0
数据结构与算法学习十一:冒泡排序、选择排序、插入排序
|
6月前
|
搜索推荐 测试技术
【数据结构常见七大排序(一)】—插入排序篇【直接插入排序】And【希尔排序】
【数据结构常见七大排序(一)】—插入排序篇【直接插入排序】And【希尔排序】
|
7月前
|
机器学习/深度学习 算法 搜索推荐
数据结构算法--2 冒泡排序,选择排序,插入排序
**基础排序算法包括冒泡排序、选择排序和插入排序。冒泡排序通过相邻元素比较交换,逐步将最大值“冒”到末尾,平均时间复杂度为O(n^2)。选择排序每次找到剩余部分的最小值与未排序部分的第一个元素交换,同样具有O(n^2)的时间复杂度。插入排序则类似玩牌,将新元素插入到已排序部分的正确位置,也是O(n^2)复杂度。这些算法适用于小规模或部分有序的数据。**
|
7月前
|
算法 搜索推荐
数据结构与算法-插入排序
数据结构与算法-插入排序
40 2
|
8月前
|
存储 搜索推荐 算法
[数据结构]————排序总结——插入排序(直接排序和希尔排序)—选择排序(选择排序和堆排序)-交换排序(冒泡排序和快速排序)—归并排序(归并排序)
[数据结构]————排序总结——插入排序(直接排序和希尔排序)—选择排序(选择排序和堆排序)-交换排序(冒泡排序和快速排序)—归并排序(归并排序)
|
8月前
|
C语言
【C语言/数据结构】排序(直接插入排序|希尔排序)
【C语言/数据结构】排序(直接插入排序|希尔排序)
61 4
|
8月前
|
搜索推荐 算法 C++
[数据结构]-玩转八大排序(一)&&插入排序&&选择排序
[数据结构]-玩转八大排序(一)&&插入排序&&选择排序
|
8月前
|
存储 算法 搜索推荐
【数据结构与算法】:插入排序与希尔排序
欢迎大家来到初阶数据结构的最后一小节:排序
【数据结构与算法】:插入排序与希尔排序
|
7月前
|
人工智能 算法 C语言
数据结构与算法——简单排序-冒泡排序、插入排序,时间复杂度下界(图示、代码、时间复杂度、定理)
数据结构与算法——简单排序-冒泡排序、插入排序,时间复杂度下界(图示、代码、时间复杂度、定理)
47 0
|
7月前
|
存储 搜索推荐
深入了解数据结构第四弹——排序(1)——插入排序和希尔排序
深入了解数据结构第四弹——排序(1)——插入排序和希尔排序
34 0

热门文章

最新文章