【算法】一文搞懂归并排序

概念

归并排序利用了分治思想,将待排序的数组范围层层划分,每次划分会得到两个大小相近的区间。当无法划分时,递归结束,自下而上进行区间合并merge操作,合并操作依次比较两个区间的元素,进而使合并后的区间有序。当进行到最后一次合并区间时,我们将会得到目标范围的有序序列。这个过程如图所示。

在这里插入图片描述

由于归并排序并不是基于交换的排序算法,因此没办法做到原地in-place排序,而是需要借助一个辅助数组,以便在合并时,可以暂时存放左侧区间的元素,这样可以将经过比较得到的元素直接放入原数组的正确位置。这也是归并排序空间复杂度的主要来源。

此外,归并排序是一种稳定的排序算法。需要注意的是,在合并过程中,左区间的元素在原数组中的位置总是位于右区间元素的左边。因此,当比较两个区间的当前元素时,如果左区间的元素小于或等于右区间的元素,应优先选择左区间的元素放入结果数组。这样可以保证相等元素的相对顺序不变,从而维护算法的稳定性。(切忌漏掉等于)

递归实现

inline void merge(int *arr, int *aux, const int left, const int mid, const int right) {
    for (int i = left; i < mid + 1; ++i) {
        aux[i] = arr[i];  // * 为避免冗余拷贝,此处先将左区间移至辅助数组
                          // * 排序时直接向arr写入数据即可,而不用最后冗余执行拷贝
    }
    int s1 = left, s2 = mid + 1, k = left;
    while (s1 <= mid && s2 <= right) {
        if (aux[s1] <= arr[s2]) { // * 请注意,为了保持排序算法稳定,这里应该是小于等于
            arr[k++] = aux[s1++];
        } else {
            arr[k++] = arr[s2++];
        }
    }
    while (s1 <= mid) {
        arr[k++] = aux[s1++];
    }
    while (s2 <= right) {
        arr[k++] = arr[s2++];
    }
}

// * 递归版本-归并排序
// * 排序区间:[left, right)
void mergeSort(int *arr, int *aux, const int left, const int right) {
    if (left >= right)
        return;
    const int mid = (left + right) / 2;
    mergeSort(arr, aux, left, mid);
    mergeSort(arr, aux, mid + 1, right);
    merge(arr, aux, left, mid, right);
}

非递归实现

// * 迭代版本-归并排序
void mergeSort2(int *arr, int n) {
    int *aux = new int[n];
    for (int step = 1; step < n; step *= 2) { // 以循环的形式,用指针控制区间划分,逐一区间进行合并,区间范围依次倍增
        for (int i = 0; i < n - 1; i += step * 2) {
            int left = i, mid = i + step - 1, right = min(i + 2 * step - 1, n - 1);
            merge(arr, aux, left, mid, right);
        }
    }
    delete[] aux;
}

复杂度分析

时间复杂度:递归划分深度为 O ( l g n ) O(lgn) O(lgn),合并每一层的时间复杂度为 O ( n ) O(n) O(n),因此总时间为 O ( n l g n ) O(nlgn) O(nlgn)
空间复杂度: O ( n ) O(n) O(n),辅助空间必不可少。

技巧应用

LCR 170. 交易逆序对的总数 - 力扣(LeetCode)

在股票交易中,如果前一天的股价高于后一天的股价,则可以认为存在一个「交易逆序对」。请设计一个程序,输入一段时间内的股票交易记录 record,返回其中存在的「交易逆序对」总数。

示例 1:
输入:record = [9, 7, 5, 4, 6]
输出:8
解释:交易中的逆序对为 (9, 7), (9, 5), (9, 4), (9, 6), (7, 5), (7, 4), (7, 6), (5, 4)

在归并排序的每一次merge操作时,我们有两个指针s1s2分别指向待合并的左右两个区间中的元素,且这两个区间的数据有两个特点,即左区间的数在原数组中一定在右区间的左侧,且左右两区间的数据各自是有序的。因此,当归并进行到arr[s1]小于arr[s2]时,假设左区间为[l, m],那么arr[s1]一定和右区间中s2及其左侧元素均构成逆序对,这个逆序对的数量为m - s1 + 1,最后我们在递归过程中将这些逆序对数目收集起来即可。

示例代码如下:

// 归并排序解法
class Solution {
private:
    int mergeSort(vector<int>& record, vector<int> &aux, int l, int r) {
        if (l >= r) return 0;
        int m = (l + r) / 2;
        // 递归
        int rtn = mergeSort(record, aux, l, m) + mergeSort(record, aux, m + 1, r);
        for (int i = l; i <= m; ++i) {
            aux[i] = record[i];
        }
        int s1 = l, s2 = m + 1, k = l;
        // 合并阶段
        while (s1 <= m && s2 <= r) {
            if (aux[s1] <= record[s2]){
                record[k++] = aux[s1++];
            } else {
                rtn += m - s1 + 1; // 统计逆序对数目
                record[k++] = record[s2++];
            }
        }
        while (s1 <= m) {
            record[k++] = aux[s1++];
        }
        while (s2 <= r) {
            record[k++] = record[s2++];
        }
        return rtn;
    }
public:
    int reversePairs(vector<int>& record) {
        if(record.empty()) return 0;
        vector<int> aux(record.size(), 0);
        return mergeSort(record, aux, 0, record.size() - 1);
    }
};

时间复杂度: O ( n l o g n ) O(nlogn) O(nlogn), 相较于暴力双循环的 O ( n 2 ) O(n^2) O(n2)有显著提升。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/676110.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

开发和渗透偷懒利器utools

目录 1.前言 1.1 工具简介 1.2 核心特性 1.3 使用场景 1.4 安装与使用 1.4.1 下载&#xff1a; 1.4.2 安装&#xff1a; 1.4.3 配置&#xff1a; 1.4.4 插件市场&#xff1a; 2.懒狗插件介绍 基本介绍 2.1 数据模拟 2.2 随机生成虚假数据 2.3 API市场 2.4 Hoppscot…

学习小心意——简单的循坏语句

for循坏 基本语法格式 for 变量 in 序列:代码块 示例代码如下 for i in range(10):print(i)#输出结果:0 1 2 3 4 5 6 7 8 9 简单案例代码如下 利用for语句遍历序列 # 遍历字符串打印每个字母 for letter in "python":print(letter)# 遍历列表并打印每个元素 a …

Spring Boot API 编写的十个最佳实践,你知道几个?

一个好的 API 不仅能提高开发效率&#xff0c;还能确保系统的安全性和稳定性。 第一部分&#xff1a;RESTful API 设计 资源名称&#xff1a;使用名词表示资源&#xff0c;比如 /users。 HTTP 方法&#xff1a;GET、POST、PUT、DELETE 分别对应查询、创建、更新和删除操作。 …

SaaS 电商设计 (十一) 那些高并发电商系统的限流方案设计

目录 一.什么是限流二.怎么做限流呢2.1 有哪些常见的系统限流算法2.1.1 固定窗口2.1.1 滑动窗口2.1.2 令牌桶2.1.3 漏桶算法 2.2 常见的限流方式2.2.1 单机限流&集群限流2.2.2 前置限流&后置限流 2.3 实际落地是怎么做的2.3.1 流量链路2.3.2 各链路限流2.3.2.1 网关层2…

【Java面试】七、SpringMvc的执行流程、SpringBoot自动装配原理

文章目录 1、SpringMVC的执行流程1.1 视图阶段1.2 前后端分离阶段 2、SpringBoot自动配置原理3、框架常用的注解3.1 Spring的注解3.2 SpringMvc的注解3.3 SpringBoot的注解 4、面试 1、SpringMVC的执行流程 1.1 视图阶段 旧项目中&#xff0c;未前后端分离时&#xff0c;用到…

计算机视觉与模式识别实验2-2 SIFT特征提取与匹配

文章目录 &#x1f9e1;&#x1f9e1;实验流程&#x1f9e1;&#x1f9e1;SIFT算法原理总结&#xff1a;实现SIFT特征检测和匹配通过RANSAC 实现图片拼接更换其他图片再次测试效果&#xff08;依次进行SIFT特征提取、RANSAC 拼接&#xff09; &#x1f9e1;&#x1f9e1;全部代…

(CVPRW,2024)可学习的提示:遥感领域小样本语义分割

文章目录 相关资料摘要引言方法训练基础类别新类别推理 相关资料 论文&#xff1a;Learnable Prompt for Few-Shot Semantic Segmentation in Remote Sensing Domain 代码&#xff1a;https://github.com/SteveImmanuel/OEM-Few-Shot-Learnable-Prompt 摘要 小样本分割是一项…

JMeter性能测试实现与分析分享

JMeter是由著名开源软件巨头Apache组织开发的纯Java的压力测试工具&#xff0c;它即能测试动态服务&#xff08;WebService&#xff09;&#xff0c;也能测试静态资源&#xff0c;包括Servlet服务、CGI脚本等&#xff0c;还能测试动态语言服务&#xff08;PHP、Java、ASP.NET等…

外汇天眼:总是权衡利弊,投资注定失败

投资股票的人往往会频繁地评估自己的投资结果&#xff0c;尤其是在信息时代&#xff0c;手机上随时可以查看股票行情&#xff0c;导致很多人时不时地打开行情软件&#xff0c;看一看自己的股票是涨了还是跌了&#xff0c;盈利了还是亏损了。 频繁评估结果的弊端 一、引发急躁…

JVM学习-监控工具(二)

jmap&#xff1a;导出内存映像文件&内存使用情况 基本情况 jmap(JVM Memory Map)&#xff1a;一方法获取dump文件(堆转储快照文件&#xff0c;二进制文件)&#xff0c;还可以获取目标Java进程的内存相关信息&#xff0c;包括Java堆各区域的使用情况、堆中对象的统计信息、…

Golang TCP网络编程

文章目录 网络编程介绍TCP网络编程服务器监听客户端连接服务器服务端获取连接向连接中写入数据从连接中读取数据关闭连接/监听器 简易的TCP回声服务器效果展示服务端处理逻辑客户端处理逻辑 网络编程介绍 网络编程介绍 网络编程是指通过计算机网络实现程序间通信的一种编程技术…

成功解决“ModuleNotFoundError: No Module Named ‘utils’”错误的全面指南

成功解决“ModuleNotFoundError: No Module Named ‘utils’”错误的全面指南 在Python编程中&#xff0c;遇到ModuleNotFoundError: No Module Named utils这样的错误通常意味着Python解释器无法找到名为utils的模块。这可能是由于多种原因造成的&#xff0c;比如模块确实不存…

推荐一款开源Scada,数据采集必备

软件介绍 PySCADA的核心是一个基于HTML5的HMI&#xff0c;不仅确保现代性&#xff0c;还确保与各种设备的无缝集成。该框架通过提供对广泛工业协议的支持&#xff0c;进一步巩固其功能。这些包括Modbus TCP/IP、RTU、ASCII和Binary&#xff0c;让用户可以轻松地与不同的设备和系…

党历史纪念馆3d沉浸式虚拟展厅为参观者提供了一个身临其境的革命教育世界

传统的展览方式虽然能够传达一定的信息&#xff0c;但往往难以激发参观者的深度参与和共鸣。而廉政教育VR数字化展厅通过VR虚拟现实技术的运用&#xff0c;打破了传统展览的局限性&#xff0c;为参观者提供了一个身临其境的廉政世界。 在这个廉政教育VR数字化展厅中&#xff0c…

高中数学:解三角形-大题练习

例题1 解析 第一小问 根据条件等式&#xff0c;我们发现&#xff0c;每一项都含有边&#xff0c;但是&#xff0c;不是每一项都含有角 于是&#xff0c;我们要想到用正弦定理把边换为角来解答该题 第二小问 例题2 解析 第一小问 两个等式条件&#xff0c;各个项都含有边&…

代码随想录算法训练营第十三天|239. 滑动窗口最大值、 347.前 K 个高频元素

239. 滑动窗口最大值 题目链接&#xff1a;滑动窗口最大值 文档讲解&#xff1a;代码随想录 状态&#xff1a;忘了 思路1&#xff1a;使用优先队列来维护滑动窗口中的最大值&#xff0c;确保在每次滑动时能够高效地找到最大值。时间复杂度是 O(n log k)。 题解&#xff1a; cl…

MyBatis的各种查询功能

1、查询&#xff1a; 查询的标签select必须设置属性resultType或resultMap&#xff0c;用于设置实体类和数据库表的映射关系 resultType&#xff1a;自动映射&#xff0c;用于属性名和表中字段名一致的情况 resultMap&#xff1a;自定义映射&#xff0c;用于一对多或多对一或…

每日一练——分糖果2

1103. 分糖果 II - 力扣&#xff08;LeetCode&#xff09; /*** Note: The returned array must be malloced, assume caller calls free().*/ int* distributeCandies(int candies, int num_people, int* returnSize) {int num 0;int* arr (int*)malloc(sizeof(int)*num_peo…

C语言 链表经典OJ题

链表经典OJ题 移除链表元素链表的中间节点反转链表合并两个有序链表分割链表 移除链表元素 给你一个链表的头节点 head 和一个整数 val &#xff0c;请你删除链表中所有满足 Node.val val 的节点&#xff0c;并返回 新的头节点 。 示例 1&#xff1a; 输入&#xff1a;head […

贝锐蒲公英异地组网:降低建筑工地远程视频监控成本、简化运维

中联建设集团股份有限公司是一家建筑行业的施工单位&#xff0c;专注于建筑施工&#xff0c;业务涉及市政公用工程施工总承包、水利水电工程施工总承包、公路工程施工总承包、城市园林绿化专业承包等&#xff0c;在全国各地开展有多个建筑项目&#xff0c;并且项目时间周期可能…