当前位置:首页 > 力扣 > 力扣912排序题终极解法:递归分割 + 双指针合并详解

力扣912排序题终极解法:递归分割 + 双指针合并详解

5个月前 (05-12)

题目解读

给定一个整数数组,要求将其按升序排列并返回。题目通常隐含对算法时间复杂度的要求,理想情况下需实现 O(n log n) 的时间复杂度。本题看似简单,但需要选择合适的排序算法(如归并排序、快速排序等),且需避免直接调用语言内置的排序函数。提供的代码通过归并排序的分治思想解决了这一问题,通过递归分割数组并合并有序子数组完成排序。


力扣912排序题终极解法:递归分割 + 双指针合并详解 C++ 算法 力扣 数组 归并排序 双指针 第1张


解题思路与过程

采用经典的归并排序算法实现排序,主要分为以下步骤:

1.递归分割数组‌:将数组不断二分,直到子数组长度为1(即递归终止条件 l == r)。

2.合并有序子数组‌:将两个已排序的子数组合并为一个有序数组。合并时使用双指针遍历左右子数组的元素,依次选择较小的值存入临时数组 tmp,最终将 tmp 的结果覆盖原数组的对应区间。

3.临时数组的利用‌:在合并过程中,通过全局临时数组 tmp[50000] 保存合并结果,避免多次动态内存分配


代码:

class Solution {  
public:  
    int tmp[50000];  // 全局临时数组,用于合并过程中的中间存储  
    void mysort(vector<int>& nums, int l, int r) {  
        if (l == r) {  // 递归终止条件:子数组长度为1  
            return;  
        }  
        int mid = (l + r) / 2;  
          
        // 递归分割左半部分  
        mysort(nums, l, mid);  
        // 递归分割右半部分  
        mysort(nums, mid + 1, r);  
          
        int index = l;    // 合并结果的起始位置  
        int lnow = l;     // 左子数组的指针  
        int rnow = mid + 1;  // 右子数组的指针  
          
        // 合并左右子数组  
        while (lnow <= mid || rnow <= r) {  
            if (rnow > r) {  // 右子数组已遍历完,直接填充左子数组剩余元素  
                tmp[index] = nums[lnow];  
                lnow++;  
                index++;  
            } else if (lnow > mid) {  // 左子数组已遍历完,填充右子数组剩余元素  
                tmp[index] = nums[rnow];  
                rnow++;  
                index++;  
            } else {  
                // 选择左右子数组中较小者放入临时数组  
                tmp[index] = min(nums[lnow], nums[rnow]);  
                if (nums[lnow] < nums[rnow]) {  
                    lnow++;  
                } else {  
                    rnow++;  
                }  
                index++;  
            }  
        }  
        // 将临时数组结果覆盖到原数组  
        for (int i = l; i <= r; i++) {  
            nums[i] = tmp[i];  
        }  
    }  
      
    vector<int> sortArray(vector<int>& nums) {  
        mysort(nums, 0, nums.size() - 1);  
        return nums;  
    }  
};


原创内容 转载请注明出处

分享给朋友:

相关文章

力扣1137题:动态规划解泰波那契数 高效求解第N项的秘密

力扣1137题:动态规划解泰波那契数 高效求解第N项的秘密

一:重新解读题目泰波那契数列是一个充满数学趣味的递推序列:从第3项开始,每个数均为前三个数的和(即Tₙ₊₃ = Tₙ + Tₙ₊₁ + Tₙ₊₂)。当给定整数n时,需要高效计算出第n项的值。面对此类递...

力扣540题:线性扫描法如何高效定位唯一数

力扣540题:线性扫描法如何高效定位唯一数

题目重解一个严格递增的有序数组中,除某个元素外,其余每个元素均出现两次。这个看似简单的条件背后隐藏着巧妙的规律——单一元素会打破数组的"成对对称性"。题目要求以O(log n)时间...

【洛谷1184题解析】用C++高效解决地点匹配问题(附代码与解题思路)

【洛谷1184题解析】用C++高效解决地点匹配问题(附代码与解题思路)

一、题目解读洛谷1184题要求处理一组地点列表与行程记录,统计其中匹配的天数。题目难点在于高效处理带有空格的字符串输入,以及快速判断每日行程是否在高手可去地点集合中。需要兼顾输入格式解析与算法效率。二...

洛谷1220题解:动态规划与区间DP优化解法(附代码注释)

洛谷1220题解:动态规划与区间DP优化解法(附代码注释)

一、题目解读洛谷1220题要求计算在n个位置放置灯的情况下,通过关闭连续区间灯并移动至区间端点,使得总耗电量最小。需考虑灯的功率与位置差异,设计高效的算法求解最优策略。二、解题思路1. 动态规划 +...

牛客NC67题解:汉诺塔递归算法与解题步骤

牛客NC67题解:汉诺塔递归算法与解题步骤

一、题目解读牛客NC67题要求解决汉诺塔问题,这是一个经典的递归算法题目。题目给定整数n,代表汉诺塔中的盘子数量,需要输出将n个盘子从起始柱移动到目标柱的所有步骤。汉诺塔问题规则为:每次只能移动一个盘...

力扣148题:合并两个有序链表的归并排序解法(递归分治优化详解)

力扣148题:合并两个有序链表的归并排序解法(递归分治优化详解)

一、题目解读力扣148题要求合并两个有序链表,返回一个有序链表。题目核心在于处理链表节点指针的遍历与合并逻辑,需确保合并后的链表保持升序。输入为两个已排序的链表头节点,输出为合并后的链表头节点。例如,...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。