排序算法——归并排序(递归与非递归)

简介: 排序算法——归并排序(递归与非递归)

归并排序

以升序为例


基本思想

  • 归并排序是建立在归并操作上的一种有效的排序算法,该算法是采用分治法的一个非常典型的应用:将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。
  • 如果对两个有序序列的归并操作还不太熟悉,建议先看看合并两个有序链表

核心步骤

  • 由上图我们可以看到,归并排序首先要对待排序列不断二分,直到分成不可分割的子序列(即只有一个元素的序列,相当于有序)
  • 然后,再对有序的子序列不断进行归并操作,最后得到完全有序的序列。
  • 归并排序有递归和非递归两种写法,接下来我们来讨论如何实现的具体细节:

递归写法

  • 首先我们要注意,在进行归并操作时,为了防止原序列的元素被覆盖而导致排序错误,我们需要向内存申请一块空间用来临时存放合并的序列,同时,由于归并的此时不止一次,为防止多次申请内存而导致效率不高,我们直接向内存申请一块和原序列大小相等的空间
void MergeSort(int *nums, int numsSize)
{
  int* temp = (int*)malloc(sizeof(int) * numsSize);
  …………;
  free(temp);                           
}
  • 同时,归并排序在进行归并操作时需要知道每个子序列的区间,由于递归参数的限制,我们需要再定义一个子函数MergeSort(),并对这个子函数递归
void _MergeSort(int *nums, int *temp, int left, int right)
{
    …………;
}
void MergeSort(int *nums, int numsSize)
{
  int* temp = (int*)malloc(sizeof(int) * numsSize);
  _MergeSort(nums, temp, 0, numsSize - 1);
  free(temp);                           
}
  • 易知,当子序列长度为1时,就可以不再进行二分
void _MergeSort(int *nums, int *temp, int left, int right)
{
  if (left >= right)
    return;
  …………;
}
  • 对待排序列的左半部分和右半部分不断递归分割
void _MergeSort(int *nums, int *temp, int left, int right)
{
  if (left >= right)
    return;
  int mid = (right - left) / 2 + left;
  _MergeSort(nums, temp, left, mid);
  _MergeSort(nums, temp, mid + 1, right);
    ……………;
}
  • 接下来,就是对两个有序序列的合并操作
  • 注:可以走到合并这一步说明待合并的两个序列[left,mid]和[mid + 1,right]是有序的,存在两种情况:
  • 情况一:例如序列{9,2},进入函数_MergeSort()后,其子序列是单个数字,满足left >= right的条件,直接退出递归,开始合并。
  • 情况二:例如序列{9,2,5,4},进入函数_MergeSort()后,子序列{9,2}递归,合并后退出递归,然后子序列{5,4}递归,合并,退出递归,最后就变成了两个有序序列{2,9}和{4,5}的合并
  • 建议对递归不是很清楚的小伙伴可以尝试画画递归展开图,这对了解递归逻辑大有裨益。
void _MergeSort(int *nums, int *temp, int left, int right)
{
  if (left >= right)
    return;
    //递归分割
  int mid = (right - left) / 2 + left;
  _MergeSort(nums, temp, left, mid);
  _MergeSort(nums, temp, mid + 1, right);
  int begin1 = left, end1 = mid;
  int begin2 = mid + 1, end2 = right;
  int index = left;
    //归并
  while (begin1 <= end1 && begin2 <= end2)
  {
    if (nums[begin1] > nums[begin2])
      temp[index++] = nums[begin2++];
    else
      temp[index++] = nums[begin1++];
  }
  while (begin1 <= end1)
    temp[index++] = nums[begin1++];
  while (begin2 <= end2)
    temp[index++] = nums[begin2++];
    //将temp暂时存储的数据覆盖待排序列nums原有位置的数据,实现待排序列区间有序
  for (int i = left; i <= right; i++)
    nums[i] = temp[i];
}

实现代码

void _MergeSort(int *nums, int *temp, int left, int right)
{
  if (left >= right)
    return;
    //递归分割
  int mid = (right - left) / 2 + left;
  _MergeSort(nums, temp, left, mid);
  _MergeSort(nums, temp, mid + 1, right);
  int begin1 = left, end1 = mid;
  int begin2 = mid + 1, end2 = right;
  int index = left;
    //归并
  while (begin1 <= end1 && begin2 <= end2)
  {
    if (nums[begin1] > nums[begin2])
      temp[index++] = nums[begin2++];
    else
      temp[index++] = nums[begin1++];
  }
  while (begin1 <= end1)
    temp[index++] = nums[begin1++];
  while (begin2 <= end2)
    temp[index++] = nums[begin2++];
    //将temp暂时存储的数据覆盖待排序列nums原有位置的数据(拷贝回去),实现待排序列区间有序
  for (int i = left; i <= right; i++)
    nums[i] = temp[i];
}
void MergeSort(int *nums, int numsSize)
{
  int* temp = (int*)malloc(sizeof(int) * numsSize);
  _MergeSort(nums, temp, 0, numsSize - 1);
  free(temp);                           
}

非递归

  • 我们可以直接用循环解决问题,如图所示:

  • 由上面的递归分析可以知道,两个单个数字可以直接合并成一个有序序列。因此我们定义gap,表示每次合并的两个序列长度为gap,gap从1递增,直到不能满足条件gap < numsSize,然后就进行和递归相同的合并操作就可以了
void MergeSort(int* nums, int numsSize)
{
  int* temp = (int*)malloc(sizeof(int) * numsSize);
  int gap = 1;
  while (gap < numsSize)
  {
        /*
          因为每次是对两个有序序列进行合并
          因此每次合并过后i应该跳过两个序列长度
        */
    for (int i = 0; i < numsSize; i += 2 * gap)
    {
      int begin1 = i, end1 = i + gap - 1;
      int begin2 = i + gap, end2 = i + 2 * gap - 1;
      int index = begin1;
            //归并
      while (begin1 <= end1 && begin2 <= end2)
      {
        if (nums[begin1] < nums[begin2])
          temp[index++] = nums[begin1++];
        else
          temp[index++] = nums[begin2++];
      }
      while(begin1 <= end1)
        temp[index++] = nums[begin1++];
      while(begin2 <= end2)
        temp[index++] = nums[begin2++];
            //将temp暂时存储的数据覆盖待排序列nums原有位置的数据(拷贝回去),实现待排序列区间有序
      for (int j = i; j <= end2; j++)
        nums[j] = temp[j];
    }
    gap *= 2;
  }
  free(temp);
}

处理边界情况

  • 看起来好像很简单,但上面的代码仍存在些许bug,仍需要我们谨慎处理
  • 我们来看一个情况,如果给的待排数组是{5,4,3,2,9,7,1,6,8}

  • 如果给的待排数组是{5,4,3,2,9,7

int begin1 = i, end1 = i + gap - 1;
int begin2 = i + gap, end2 = i + 2 * gap - 1;
int index = begin1;
/*
  如果右半区间不存在,只有左半区间
  说明待合并的只有一个区间
  显然没有合并的必要,直接退出合并循环即可
*/
if (begin2 >= numsSize)
    break;
//如果右半区间算多了,那么对end2进行修正
if (end2 >= numsSize)
    end2 = numsSize - 1;

实现代码

void MergeSort(int* nums, int numsSize)
{
  int* temp = (int*)malloc(sizeof(int) * numsSize);
  int gap = 1;
  while (gap < numsSize)
  {
        /*
          因为每次是对两个有序序列进行合并
          因此每次合并过后i应该跳过两个序列长度
        */
    for (int i = 0; i < numsSize; i += 2 * gap)
    {
      int begin1 = i, end1 = i + gap - 1;
            int begin2 = i + gap, end2 = i + 2 * gap - 1;
            int index = begin1;
            /*
                如果右半区间不存在,只有左半区间
                说明待合并的只有一个区间
                显然没有合并的必要,直接退出合并循环即可
            */
            if (begin2 >= numsSize)
                break;
            //如果右半区间算多了,那么对end2进行修正
            if (end2 >= numsSize)
                end2 = numsSize - 1;
       //归并
      while (begin1 <= end1 && begin2 <= end2)
      {
        if (nums[begin1] < nums[begin2])
          temp[index++] = nums[begin1++];
        else
          temp[index++] = nums[begin2++];
      }
      while(begin1 <= end1)
        temp[index++] = nums[begin1++];
      while(begin2 <= end2)
        temp[index++] = nums[begin2++];
            //将temp暂时存储的数据覆盖待排序列nums原有位置的数据(拷贝回去),实现待排序列区间有序
      for (int j = i; j <= end2; j++)
        nums[j] = temp[j];
    }
    gap *= 2;
  }
  free(temp);
}

时间复杂度

  • 易得,合并两个有序序列的时间复杂度为O(N)
  • 由于是对待排序列的不断二分,知道分割为不可分割的子序列,因此这一过程的时间复杂度为O(log2N)
  • 因此归并排序的时间复杂度为O(NlogN)
相关文章
|
2月前
|
算法 Python
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果;贪心算法在每一步选择局部最优解,追求全局最优;动态规划通过保存子问题的解,避免重复计算,确保全局最优。这三种算法各具特色,适用于不同类型的问题,合理选择能显著提升编程效率。
70 2
|
3月前
|
算法 搜索推荐 Shell
数据结构与算法学习十二:希尔排序、快速排序(递归、好理解)、归并排序(递归、难理解)
这篇文章介绍了希尔排序、快速排序和归并排序三种排序算法的基本概念、实现思路、代码实现及其测试结果。
62 1
|
3月前
|
算法 定位技术
数据结构与算法学习九:学习递归。递归的经典实例:打印问题、阶乘问题、递归-迷宫问题、八皇后问题
本文详细介绍了递归的概念、重要规则、形式,并展示了递归在解决打印问题、阶乘问题、迷宫问题和八皇后问题等经典实例中的应用。
61 0
|
3月前
|
存储 搜索推荐 算法
【排序算法(二)】——冒泡排序、快速排序和归并排序—>深层解析
【排序算法(二)】——冒泡排序、快速排序和归并排序—>深层解析
|
3月前
|
存储 算法 搜索推荐
算法进阶之路:Python 归并排序深度剖析,让数据排序变得艺术起来!
算法进阶之路:Python 归并排序深度剖析,让数据排序变得艺术起来!
87 0
|
3月前
|
搜索推荐 Java Go
深入了解归并排序算法
深入了解归并排序算法
52 0
|
5月前
|
算法 搜索推荐 Java
算法实战:手写归并排序,让复杂排序变简单!
归并排序是一种基于“分治法”的经典算法,通过递归分割和合并数组,实现O(n log n)的高效排序。本文将通过Java手写代码,详细讲解归并排序的原理及实现,帮助你快速掌握这一实用算法。
56 0
|
5月前
|
算法
【算法】递归、搜索与回溯——汉诺塔
【算法】递归、搜索与回溯——汉诺塔
|
5月前
|
数据采集 搜索推荐 算法
【高手进阶】Java排序算法:从零到精通——揭秘冒泡、快速、归并排序的原理与实战应用,让你的代码效率飙升!
【8月更文挑战第21天】Java排序算法是编程基础的重要部分,在算法设计与分析及实际开发中不可或缺。本文介绍内部排序算法,包括简单的冒泡排序及其逐步优化至高效的快速排序和稳定的归并排序,并提供了每种算法的Java实现示例。此外,还探讨了排序算法在电子商务、搜索引擎和数据分析等领域的广泛应用,帮助读者更好地理解和应用这些算法。
51 0
|
6月前
|
算法 Python
python中算法递归错误(Recursion Errors)
【7月更文挑战第18天】
101 1