归并排序三种常见写法

算法思路

归并排序是一种分治算法:首先将数组分成两半,然后对每一半进行归并排序,最后将两个有序的子数组合并,以得到最终的排序数组。为了简洁下面代码中会调用 STL 的 i n p l a c e _ m e r g e inplace\_merge inplace_merge 方法,这个方法的作用正是将两个连续的有序区间合并为一个有序区间,当然也可以自己按合并有序链表的思路写一个 m e r g e merge merge 方法~


a. 递归

自顶向下的递归排序

cpp 复制代码
void merge_sort(vector<int> &li, int s, int e) {//li[s,e)为待排序的子数组
    if (e - s <= 1)
        return;
    int mid = (s + e) / 2;
    merge_sort(li, s, mid);
    merge_sort(li, mid, e);
    inplace_merge(li.begin() + s, li.begin() + mid, li.begin() + e);//合并有序区间[s,mid)和[mid,e)
}

b. 栈模拟递归

用一个栈模拟递归排序中区间处理的过程,注意一个区间可能会在栈中出现两次:第一次其两个子区间还未排序,第二次其两个子区间已经有序,所以需要有一个额外的标志位来区分一个区间在栈中的这两种状态。

cpp 复制代码
void merge_sort_stack(vector<int> &li, int s, int e) {//li[s,e)为待排序的子数组
    stack<tuple<int, int, int>> st;
    st.emplace(s, e, 0);
    while (!st.empty()) {
        auto [l, r, tag] = st.top();//当前区间为[l,r) , 标志位为tag
        st.pop();
        if (tag == 0) {//当前区间第一次出现
            if (r - l <= 1)
                continue;
            int mid = (l + r) / 2;
            st.emplace(l, r, 1);
            st.emplace(l, mid, 0);
            st.emplace(mid, r, 0);
        } else {//当前区间第二次出现
            int mid = (l + r) / 2;
            inplace_merge(li.begin() + l, li.begin() + mid, li.begin() + r);//合并有序区间[l,mid)和[mid,r)
        }
    }
}

c. 递推

自底向上的递推:首先两两一组的合并长为 2 0 2^0 20 的子数组,再两两一组的合并长为 2 1 2^1 21 的子数组, ⋯ \cdots ⋯ ,直到 2 k 2^k 2k 不小于数组长度时递推结束。注意每一轮合并中最后一组的右边的子数组的长度可能小于 2 k 2^k 2k。

cpp 复制代码
void merge_sort(vector<int> &li, int s, int e) {//li[s,e)为待排序的子数组
    int n = e - s;
    for (int len = 1; len < n; len *= 2)//合并的子数组的长度len不超过n
        for (int i = s; i + len < e; i += len * 2)//逐对枚举两个相邻的子数组
            inplace_merge(li.begin() + i, li.begin() + i + len, li.begin() + min(e, i + len * 2));//合并有序区间[i,i+len)和[i+len,min(e,i+len*2))
}
相关推荐
闻缺陷则喜何志丹12 分钟前
【贪心 字典序 回文 最长公共前缀】LeetCode3734. 大于目标字符串的最小字典序回文排列|分数未知
c++·算法·力扣·贪心·字典序·回文·最长公共前缀
weixin_5142218531 分钟前
FDTD代码学习-1
学习·算法·lumerical·fdtd
AI柠檬1 小时前
机器学习:数据集的划分
人工智能·算法·机器学习
_OP_CHEN1 小时前
C++进阶:(四)set系列容器的全面指南
开发语言·c++·stl·set·multiset·关联式容器·setoj题
让我们一起加油好吗2 小时前
【数论】裴蜀定理与扩展欧几里得算法 (exgcd)
算法·数论·裴蜀定理·扩展欧几里得算法·逆元
Geo_V2 小时前
提示词工程
人工智能·python·算法·ai
侯小啾2 小时前
【22】C语言 - 二维数组详解
c语言·数据结构·算法
qq_479875432 小时前
Linux time function in C/C++【2】
linux·c语言·c++
TL滕2 小时前
从0开始学算法——第一天(如何高效学习算法)
数据结构·笔记·学习·算法
傻童:CPU2 小时前
DFS迷宫问题
算法·深度优先