剑指offer ---- 数组中的逆序对数目

本文介绍了一种高效计算数组中逆序对总数的方法。通过改进归并排序算法,实现时间复杂度为O(nlogn)的逆序对计数方案,并给出具体实现代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

对于逆序对的问题,首先最容易想到的就是暴力穷举的方法。然而在题目中n的取值非常大,遍历的时间复杂度为O(n^2), 可见时间复杂度还是很高的,那么怎么才能解决这个问题呢?

我们可以根据改进一下归并排序的算法,先回忆一下归并排序的基本思想:我们把一个数组分成两部分,然后对这两部分递归的进行归并排序,然后对归并排序好的两部分进行归并,所以归并的时间复杂度为O(nlogn). 因为我们想统计数组中的逆序对数目,那么我们可以先将数组分成两部分,然后对两个数组递归统计逆序对的数目,然后统计出来两个数组间的逆序对数目(即左边数组中的数大于右边数组中的数的总对数)。根据以上分析我们的代码如下:

import math
class Solution:
    def InversePairs(self, data):
        # write code here
        count = 0
        if len(data)==1:
            return 0
        if len(data) == 2:
            if data[0]>data[1]:
                return 1
            else:
                return 0
        if len(data) > 2:
            mid = math.floor((len(data)-1)//2)
            data_l = data[:mid]
            data_r = data[mid:]
            data_sorted_l = sorted(data_l)
            data_sorted_r = sorted(data_r)
            for i in range(len(data_sorted_r)):
                for j in range(len(data_sorted_l)):
                    if data_sorted_l[j] > data_sorted_r[i]:
                        count += 1
        return (count + self.InversePairs(data_l)+self.InversePairs(data_r))%1000000007
if __name__ == '__main__':
    s = Solution()
    data1 = [364,637,341,406,747,995,234,971,571,219,993,407,416,366,315,301,601,650,418,355,460,505,360,965,516,648,727,667,465,849,455,181,486,149,588,233,144,174,557,67,746,550,474,162,268,142,463,221,882,576,604,739,288,569,256,936,275,401,497,82,935,983,583,523,697,478,147,795,380,973,958,115,773,870,259,655,446,863,735,784,3,671,433,630,425,930,64,266,235,187,284,665,874,80,45,848,38,811,267,575]
    print (s.InversePairs(data1))

 

 软件使用说明: 1,请以管理员身份运行本软件并关闭防火墙杀毒软件,软件会尝试将路由器与电脑连接的网卡以外的所有连接都禁用以免软件识别错误ip导致机,当然也有可能软件会禁用失败,建议你自己手动禁用,完机在启用,. 2,斐讯k1,k1s,k2全自动方案为全自动模式[适用于目前斐讯k1,k1s,k2出的所有固件版本] 小米路由方案需要你开启ssh权限才能使用  通用方案需要你自己开启路由器telnet或ssh然后设置路由器telnet或ssh地址密码,剩下的交给软件去完成. 3,默认r928的breed版本[软件包含所有版本的breed文件,离线] 4,不需要使用环境再安装telnet组件.也不需要设置输法. 5,特别说明,breed的时候请确定自己的路由器型号及可breed版本,以免成砖. 6,如果提示登录telnet或ssh失败而你又确定路由器telnet或ssh服务是开启的,请关闭软件重试 提供路由器免拆机开telnet及ssh方法邮件发送格式:目前全自动就k1一个方案,欢迎提供各种路由器免拆机开telnet及ssh方法,直接发我邮箱:306716509@qq.com,我有时间会集成到软件中. 路由器型号及版本: 默认telnet或ssh用户名及密码: 默认telnet或ssh ip地址: 可breed版本: 免拆机开启telnet或ssh服务方法: 2.5版本更新日志: 添加支持k2-v22.3.15.128版本,添加机完成后关闭telnet服务[仅针对v22.3.15.128版本,其他版本请勿勾选此功能否则可能会出现未知问题],添加ssh手选方案,注意不支持v21.4.6.12版本[此版本删除wget与tftp程序无法机,只能降级v21.4.6.10以下或升级成v22.3.15.128版本才可],特别注意事项,假如你先使用路由器breed助手在同一台电脑上过v22.3.15.128版本固件,在老版本可能需要重启电脑或清除ie浏览器缓存才行。 2.6版本更新日志: 修改k1,k1s,k2新版本使用升级开启ssh官方测试固件v21.99.99.99机方案支持v22.3.15到v22.4.2.9版本固件。添加启动程序先清除浏览器缓存。 原文件md5校验值:建议大家下载文件后使用hash_1.0.4校验一下路由器breed Web控制台通用版.exe文件是否跟微信:huzibbs-it 上面的路由器breed助手文章里面的md5值是否一致,如果不一致建议不要使用,可能会被注了病毒及盗号类木马程序 File: 路由器breed Web控制台通用版v2.6.exe Size: 786432 bytes File Version: 2.06 Modified: 2016年8月24日, 23:27:06 MD5: 54E501385EE96F3C34D137111361A397 SHA1: 9932E3C55865CFBF46781ED3401385269F4A8421 CRC32: BD2A4DC2
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值