插入排序 ( 直接插入排序 && 希尔排序 )

简介: 插入排序 ( 直接插入排序 && 希尔排序 )
1、直接插入排序

🔑 核心思想 🔑

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

实际中我们玩扑克牌时,就用了插入排序的思想

❗ 过程:❕

当插入第 i(i>=1) 个元素时,前面的 array[0], array[1], … , array[i-1] 已经排好序,此时用 array[i] 的排序码与 array[i-1], array[i-2],… 的排序码顺序进行比较,找到插入位置即将 array[i] 插入,原来位置上的元素顺序后移

❗ 直接插入排序的特性总结:❕

  1️⃣ 元素集合越接近有序,直接插入排序算法的时间效率越高

  2️⃣ 时间复杂度:O(N^2)

  3️⃣ 空间复杂度:O(1),它是一种稳定的排序算法

  4️⃣ 稳定性:稳定

❗ 动图演示:❕

🧿 实现代码 :

void InserSort(int* a, int n)
{
  //多趟控制
  int i = 0;
  for (i = 0; i < n - 1; i++)
  {
    //单趟控制
    int end = i ;
    int temp = a[end + 1];
    while (end >= 0)
    {
      //目标数小于其它数时,其它数就往后挪;大于则插入
      if (temp < a[end])
      {
        a[end + 1] = a[end];
        end--;
      }
      else
      {
        break;
      }
    }
    a[end + 1] = temp;
  }
}

❓ 插入排序的时间复杂度 ❔

  最坏的情况 - 逆序:O(N2)

  最好的情况 - 接近有序 :O(N)

2、希尔排序

希尔排序 (缩小增量排序)

🔑 核心思想 🔑

  希尔排序法又称缩小增量法。希尔排序法的基本思想是:先选定一个整数,把待排序文件中所有记录分成若干个组,所有距离为 gap 的记录分在同一组内,并对每一组内的记录进行排序。然后,取,重复上述分组和排序的工## 标题作。当到达 = 1 时,所有记录在统一组内排好序。

  人话就是:

    1️⃣ 预排序 (接近升序) - gap > 1

    2️⃣ 直接插入排序 - gap == 1

❗ 希尔排序特性总结 ❕

  1️⃣ 希尔排序是对直接插入排序的优化

  2️⃣ 当 gap > 1 时都是预排序,目的是让数组更接近于有序。当 gap == 1 时,其实就是直接插入排序,且数组已经接近有序的了。整体而言,可以达到优化的效果,我们实现后可以进行性能测试的对比

  3️⃣ 希尔排序的时间复杂度并不好计算,因为 gap 的取值方法很多,导致很难去计算,因此在好些数中给出的希尔排序的时间复杂度都不固定,官方给出的时间复杂度是 O(N1.3)

  4️⃣ 稳定性:不稳定

👁‍🗨 知识扩展

🧿 实现代码 :

代码的核心并不是一组一组的排,而是多组并排

以下只是预排序代码,还需要再调用 InsertSort 进行直接插入排序

void ShellSort(int* a, int n)
{
  int i = 0;
  int gap = 3;
  //多组并排
  for (i = 0; i < n - gap; i++)
  {
    int end = i;
    int temp = a[end + gap];
    while (end >= 0)
    {
      if (temp < a[end])
      {
        a[end + gap] = a[end];
        end -= gap;
      }
      else
      {
        break;
      }
    }
    a[end + gap] = temp;
  }
} 

❓ 对于 gap 的值写成固定的并不好 ❔

  这里只是建议

void ShellSortPro(int* a, int n)
{
  //gap > 1 预排序
  //gap == 1 直接插入排序
  int i = 0;
  //gap的初始值为n
  int gap = n;
  while (gap > 1)
  {
    //每次循环gap都在减少,直到gap变成1
    gap = gap / 3 + 1;
    //gap /= 2;
    for (i = 0; i < n - gap; i++)
    {
      int end = i;
      int temp = a[end + gap];
      while (end >= 0)
      {
        if (temp < a[end])
        {
          a[end + gap] = a[end];
          end -= gap;
        }
        else
        {
          break;
        }
      }
      a[end + gap] = temp;
    }
  }
}




相关文章
|
1天前
|
搜索推荐 算法 C语言
插入排序
插入排序是一种简单直观的排序算法,通过构建有序序列,将未排序的数据逐个插入到已排序序列中的适当位置。该算法采用in-place排序,只需常数级额外空间。示例代码展示了如何使用C语言实现插入排序,并对一个整数数组进行排序。
12 6
|
5月前
|
机器学习/深度学习 搜索推荐 算法
【C/排序算法】:直接插入排序和希尔排序
【C/排序算法】:直接插入排序和希尔排序
43 0
|
5月前
|
算法 搜索推荐 Java
插入排序就是这么容易
插入排序就是这么容易
31 0
|
6月前
直接插入排序与希尔排序
直接插入排序与希尔排序
42 2
|
6月前
|
搜索推荐 C++
C++插入排序的实现
C++插入排序的实现
|
6月前
|
存储 搜索推荐 算法
插入排序(一)——直接插入排序与希尔排序
插入排序(一)——直接插入排序与希尔排序
46 1
|
6月前
|
搜索推荐
直接插入排序和希尔排序
直接插入排序和希尔排序
67 0
|
6月前
|
搜索推荐 算法 测试技术
排序算法:插入排序(直接插入排序、希尔排序)
排序算法:插入排序(直接插入排序、希尔排序)
67 0
|
搜索推荐
17 插入排序
17 插入排序
33 0
插入排序与希尔排序
插入排序与希尔排序
48 0