排序算法之归并排序

归并排序(Merge Sort)

归并排序是一种分治算法(Divide and Conquer),用于将一个无序数组或链表按照一定的顺序排序(通常是升序)。归并排序的核心思想是:将数组分为两个子数组,分别排序后再合并起来,最终得到一个有序的数组。

有两种方法实现归并排序

第一种是迭代法(自底向上归并法) ,第二种是递归法(自顶向下归并法)

迭代法

通过增加归并排序区间,多次小区间的归并,从而实现整个数组的排序

cpp 复制代码
void sort_(int*,int,int,int);
void merge_sort1(int* nums,int length)
{ 
    int* temp=new int[length];    
    for(int i=1;i<length;i*=2)       //不断增加归并区间i
    {   
        int index=0;                 //归并区间距离改变 从头开始迭代 
        
        /*可以分配2个i区间*/
        while(index+2*i<=length)      
        {
             index+=2*i;               
             sort_(nums,temp,index-2*i,index-i,index); //2*i的区间 归并左i区间和右i区间
        }

        /不足以分配2个i区间/
        if(index+i<length)  sort_(nums,temp,index,index+i,length);
    }  
    delete []temp; 
}
void sort_(int* nums,int* temp,int start,int mid,int end) //头闭尾开
{
    cout<<"归并的区间为"<<start<<" "<<mid<<" "<<end<<endl;
    int l=start;
    int r=mid;
    int index=0;
    while(index<end-start)
    {    
        if(l<mid&&(r==end||nums[l]<nums[r]))  //当左边小于右边或者右区间没有数据时
        {
            temp[index++]=nums[l++];   
        }  
        else temp[index++]=nums[r++];
    }  
    
    /*拷贝数据*/
    for(int i=0;i<end-start;i++)
    {
        nums[i+start]=temp[i];       
    }
}

通过改变归并区间大小不断的扫描整个数组

递归法

cpp 复制代码
void sort_(int*,int,int,int)
void merge_sort2(int* nums,int start,int end)
{
    if(end-start<2)  return;  //只有一个元素则退出递归
    int mid=(start+end)/2;
    merge_sort2(nums,start,mid);   //递归进入左半区间
    merge_sort2(nums,mid,end);     //递归进入右半区间
    sort_(nums,start,mid,end);     //归并当前的左右区间
}

void sort_(int* nums,int start,int mid,int end) //头闭尾开
{
    cout<<"归并的区间为"<<start<<" "<<mid<<" "<<end<<endl;
    int l=start;
    int r=mid;
    int index=0;
    int* temp=new int[end-start];
    while(index<end-start)
    {    
        if(l<mid&&(r==end||nums[l]<nums[r]))  //当左边小于右边或者右区间没有数据时
        {
            temp[index++]=nums[l++];   
        }  
        else temp[index++]=nums[r++];
    }  
    
    /*拷贝数据*/
    for(int i=0;i<end-start;i++)
    {
        nums[i+start]=temp[i];       
    }
    delete[]temp;
}

可以看出它是不断的递归进入自己的左右区间,递归到左右只有一个元素然后再向上合并

两种方法的时间复杂度都为O(nlog⁡n)

这里的迭代法的空间复杂度为O(n),递归法空间复杂度为 O(nlog⁡n),你可以传一个临时数组以达到O(n)并且可以减少new和delete带来的开销

进阶版:实现对链表的归并排序

相关推荐
IT大白鼠3 小时前
AIGC性能的关键瓶颈:算力、数据、算法三者如何互相制约?
算法·aigc
白雪茫茫3 小时前
监督学习、半监督学习、无监督学习算法详解
python·学习·算法·ai
FengyunSky3 小时前
浅析 空间频率响应 SFR 计算
算法
树下水月4 小时前
PHP 一种改良版的雪花算法
算法·php·dreamweaver
一只数据集4 小时前
全尺寸人形机器人灵巧手力觉触觉数据集-2908条ROSbag数据覆盖14大应用场景深度解析
大数据·人工智能·算法·机器人
罗西的思考5 小时前
【GUI-Agent】阿里通义MAI-UI 代码阅读(2)--- 实现
人工智能·算法·机器学习
刀法如飞6 小时前
TypeScript 数组去重的 20 种实现方式,哪一种你还不知道?
前端·javascript·算法
sali-tec7 小时前
C# 基于OpenCv的视觉工作流-章66-直线夹角
图像处理·人工智能·opencv·算法·计算机视觉
AC赳赳老秦7 小时前
接口测试自动化:用 OpenClaw 对接 Postman,实现批量回归测试、测试报告自动生成与推送
java·人工智能·python·算法·elasticsearch·deepseek·openclaw
_风满楼8 小时前
TDD实战-会议室冲突检测的红绿重构循环
前端·javascript·算法