算法进阶——数组中的逆序对

news2025/2/26 15:17:30

题目


在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。输入一个数组,求出这个数组中的逆序对的总数P。并将P对1000000007取模的结果输出。 即输出P mod 1000000007

数据范围:对于 50% 的数据, size≤104
对于 100% 的数据, size≤105
数组中所有数字的值满足 0≤val≤109

要求:空间复杂度O(n),时间复杂度O(nlogn)

输入描述:
题目保证输入的数组中没有的相同的数字

示例1

输入:
[1,2,3,4,5,6,7,0]
返回值:
7

示例2

输入:
[1,2,3]
返回值:
0

思路


这题可以用归并统计法,也就是在归并排序的同时进行统计。

先了解归并排序算法,主要思想是先分后并:

  • 将数组分为两个子数组,两个子数组分为四个子数组,依次向下分,直到数组不能再分为止。
  • 从最小的数组按照顺序合并,从小到大或从大到小,依次向上合并,最后得到合并完的顺序数组。

归并统计法,关键点在于合并环节,在合并数组的时候,当发现右边的小于左边的时候,此时可以直接求出当前产生的逆序对的个数。

举个例子:
在合并 {4 ,5} {1 , 2} 的时候,首先我们判断 1 < 4,我们即可统计出逆序对为2,为什么呢?这利用了数组的部分有序性。因为我们知道 {4 ,5} 这个数组必然是有序的,因为是合并上来的。此时当 1比4小的时候,证明4以后的数也都比1大,此时就构成了从4开始到 {4,5}这个数组结束,这么多个逆序对(2个),此时利用一个临时数组,将1存放起来,接着比较2和4的大小,同样可以得到有2个逆序对,于是将2也放进临时数组中,此时右边数组已经完全没有元素了,则将左边剩余的元素全部放进临时元素中,最后将临时数组中的元素放进原数组对应的位置。

可以看到下面这张图:

解答代码


class Solution {
public:
    /**
     * @param nums int整型vector 
     * @return int整型
     */
    int InversePairs(vector<int>& nums) {
        // write code here
        int res = 0;
        vector<int> tmp(nums.size());
        MergeSort(nums, tmp, 0, nums.size() - 1, res);
        return res;
    }

    void MergeSort(vector<int>& nums, vector<int>& tmp, int left, int right, int& res) {
        // 递归结束
        if (left >= right) {
            return;
        }
        // 中心点
        int mid = left + (right - left) / 2;
        // 归并
        MergeSort(nums, tmp, left, mid, res);
        MergeSort(nums, tmp, mid + 1, right, res);
        Merge(nums, tmp, left, mid, right, res);
    }

    void Merge(vector<int>& nums, vector<int>& tmp, int left, int mid, int right, int& res) {
        int i = left; // 左子数组下标起点
        int j = mid + 1; // 右子数组下标起点
        int k = 0; //临时数组的下标起点

        while (i <= mid && j <= right) {
            if (nums[i] < nums[j]) {
                // 左子数组元素当前元素较小,无逆序,只进行排序
                tmp[k++] = nums[i++];
            } else {
                // 左子数组元素当前元素较大,排序同时记录逆序数
                tmp[k++] = nums[j++];
                res += mid + 1 - i;
                res %= 1000000007; // 应题目要求
            }
        }

        // 子数组中剩下元素全部存入临时数组
        while (i <= mid) {
            tmp[k++] = nums[i++];
        }
        while (j <= right) {
            tmp[k++] = nums[j++];
        }
        // 完成排序
        for (int i = left, j = 0; i <= right; i++, j++) {
            nums[i] = tmp[j];
        }
    }
};

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux系统CH347应用—SPI功能

Linux/安卓系统使用CH347转接SPI功能有三种应用方式&#xff1a; 1. 使用CH34X_MPHSI_Master总线驱动为系统扩展原生SPI Master&#xff0c;此方式无需进行单独的应用层编程&#xff1b; 2. 使用CH341PAR_LINUX字符设备驱动&#xff0c;此方式需要配合使用厂商提供的库文件&a…

js创建 ajax 过程

目录 前言&#xff1a;AJAX 技术的重要性 详解&#xff1a;创建 AJAX 请求的步骤 1. 创建 XMLHttpRequest 对象 2. 配置请求 3. 处理响应 4. 发送请求 5. 处理异步请求 解析&#xff1a;AJAX 请求的重要性和限制 总结&#xff1a; 前言&#xff1a;AJAX 技术的重要性 …

2019年亚太杯APMCM数学建模大赛B题区域经济活力及其影响因素的分析与决策求解全过程文档及程序

2019年亚太杯APMCM数学建模大赛 B题 区域经济活力及其影响因素的分析与决策 原题再现 区域&#xff08;或城市或省级&#xff09;经济活力是区域综合竞争力的重要组成部分。近年来&#xff0c;为了提高经济活力&#xff0c;一些地区推出了许多刺激经济活力的优惠政策&#xf…

分布式锁的基本原理和实现以及synchronized底层原理

1.1Synchronized Synchronized的重点级锁&#xff0c;底层是基于锁监督器&#xff08;Monitor&#xff09;来实现&#xff0c;简单来说就是锁对象头会指向一个锁监督器&#xff0c;而在监督器中则会记录一些信息&#xff0c;比如&#xff1a; _owner:持有锁的线程_recursion…

自动驾驶之—车道线感知

零、前言 &#xff1a; 最近在学习自动驾驶方向的东西&#xff0c;简单整理一些学习笔记&#xff0c;学习过程中发现宝藏up 手写AI 一、视觉系统坐标系 视觉系统一共有四个坐标系&#xff1a;像素平面坐标系&#xff08;u,v&#xff09;、图像坐标系&#xff08;x,y&#xff09…

华泰证券:达达集团(DADA)3Q23业绩前瞻:短期业绩承压

来源&#xff1a;猛兽财经 作者&#xff1a;猛兽财经 猛兽财经获悉&#xff0c;考虑到宏观实物消费恢复相对较弱&#xff0c;华泰证券发布关于达达集团&#xff08;DADA&#xff09;达达集团&#xff08;DADA&#xff09;3Q23业绩前瞻:短期业绩承压的研报。 华泰证券在研报中预…

【Linux】【驱动】设备树中设备节点的挂载

【Linux】【驱动】设备树中设备节点的挂载 代码操作脚本Linux中的操作下位机中的操作指令 代码 设备树对应的文件是100ask_imx6ull_mini.dtb 所以需要在根节点上增加相关的测试代码 我们修改的就是hi如下的代码部分 增加测试节点 test1:test1{#addrsee-cells < 1 >;#s…

C算法:递归算法求a的n次方

需求&#xff1a; 用递归算法写一个函数&#xff0c;实现a的n次方。 代码实现&#xff1a; #include <stdio.h> #include <stdlib.h> int nndata(int a,int n) {if(n<1){printf("please input numdata(>1) !\n");exit(-1);}if(n1){return a;}els…

基于Java的校园办公室报修管理系统设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言具体实现截图论文参考详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09; 代码参考数据库参考源码获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作者&am…

写一个JAVA逻辑的手动计算方法

逻辑&的手动实现 直接上代码吧计算结果 直接上代码吧 public class TestStream1023 {Testpublic void testPareCom() {String res testPareCom(12, 5, 8);System.out.println("逻辑&计算的结果二进制:["res"]");System.out.println("二级制…

Ubuntu22.04安装,SSH无法连接

Ubuntu初始化安装后&#xff0c;系统默认不允许root通过ssh连接&#xff0c;因此需要完成三个设置 1.修改ssh配置文件 vim /etc/ssh/sshd_config 将PermitRootLogin注释打开&#xff0c;并将值改为yes 保存修改并退出 :wq 2.重启ssh服务 sudo service ssh restart 3.重新打…

智加科技多项成果亮相ITS World Congress 两款智能重卡计划量产

2023年10月16日-20日&#xff0c;第29届智能交通世界大会&#xff08;ITS World Congress&#xff09;在苏州国际博览中心成功举办。智能交通世界大会被誉为智能交通领域的“奥运会”&#xff0c;是智能交通领域最具影响力的综合性国际会议&#xff0c;也是智能交通业界规格最高…

蒙特卡洛树搜索(MCTS)怎么实现的?+ 上置信范围Upper Confidence Bounds(UCB)是什么?

一、算法思想 我们回想一下我们下棋时的思维——并没有在脑海里面把所有可能列出来&#xff0c;而是根据「棋感」在脑海里大致筛选出了几种「最可能」的走法&#xff0c;然后再想走了这几种走法之后对手「最可能」的走法&#xff0c;然后再想自己接下来「最可能」的走法&…

第一讲之递推与递归上篇

第一讲之递推与递归上篇 数据与算法的关系简单斐波那契递归实现指数型枚举递归实现排列型枚举递归实现组合型枚举 本专栏博客&#xff0c;根据acwing中蓝桥杯CAB组辅导课编写 数据与算法的关系 简单斐波那契 简单斐波那契 斐波那契数列的话&#xff0c;只要掌握规律&#xff0…

Danielle Foré 近日向 9to5Linux 通报了 elementary OS 7.1 的发布和全面可用性

导读Danielle For 近日向 9to5Linux 通报了 elementary OS 7.1 的发布和全面可用性&#xff0c;这是自 elementary OS 7.0 “Horus “于 2023 年 1 月发布以来&#xff0c;基于 Ubuntu 的发行版的首次重大更新。 elementary OS 7.1 引入了新的隐私功能&#xff0c;包括系统设置…

RK3568 蓝牙测试

本个章节所使用的蓝牙为E104-BT5032A,这款蓝牙芯片为免驱,只需要uart正常即可使用,理论上除了3568以外都可以使用 E104-BT5032A有几个地方需要注意,首先是他有几个管脚需要配置的,经常没去留意着三个引脚的电平配置,导致使用异常,然后就是保证uart能够正常使用即可 首先…

前端git提交后菜单的配置

1.添加元数据–》app微服务 在系统中找到自己对应的位置&#xff0c;然后点击进去找到功能用例&#xff0c;添加相应的前端路由和中文名称 2.在权限管理中的菜单管理中添加相应的菜单权限

SQL关于日期的计算合集

前言 在SQL Server中&#xff0c;时间和日期是常见的数据类型&#xff0c;也是数据处理中重要的一部分。SQL Server提供了许多内置函数&#xff0c;用于处理时间和日期数据类型。这些函数可以帮助我们执行各种常见的任务&#xff0c;例如从日期中提取特定的部分&#xff0c;计…

Photoshop 2024(ps ai beta) v25.0

Photoshop 2024是一款业界领先的图像编辑软件&#xff0c;被广泛应用于设计、摄影、插图等领域。以下是这款软件的一些主要功能和特点&#xff1a; 丰富的工具和功能。Adobe Photoshop 2024提供了丰富的工具和功能&#xff0c;可以帮助用户对图像进行编辑、修饰和优化。它支持…

Linux 挂载磁盘到指定目录

问题&#xff1a;公司分配了数据磁盘&#xff0c;但是分区也没有挂载到目录 首先 df -h 查看一下挂载点的情况 查看服务器上未挂载的磁盘 fdisk -l 注&#xff1a;图中sda、sdb &#xff08;a、b指的是硬盘的序号&#xff09; 分区操作 我们可以看到b硬盘有536G未分区&…