热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

剑指offer数组中的逆序对JavaScript

题目形容:在数组中的两个数字,假如前面一个数字大于后面的数字,则这两个数字组成一个逆序对。输入一个数组,求出这个数组中的逆序对的总数。解法1:暴力法(TLE)直接双重循环,挨个检查

题目形容:在数组中的两个数字,假如前面一个数字大于后面的数字,则这两个数字组成一个逆序对。输入一个数组,求出这个数组中的逆序对的总数。

解法 1: 暴力法(TLE)

直接双重循环,挨个检查能否为逆序对。代码实现比较简单:

/** * @param {number[]} nums * @return {number} */var reversePairs = function(nums) { let res = 0; const length = nums.length; for (let i = 0; i nums[j] && ++res; } } return res;};

时间复杂度是O(N^2)。在 leetcode 上会 TLE,无法通过(毕竟这是道标注「困难」的题目)。

解法 2: 归并排序(正确解法)

这题的正确解法是要借助归并排序的思路,在归并的过程中,快速统计逆序对。这种解法比较难想到,但是应用归并排序的题目真的不多,所以这题很有研究和收藏意义。

核心的处理逻辑都封装在 findInversePairNum 函数中。它的职能就是统计数组arr[start, end]范围中的逆序对,并且统计完后,arr[start, end]范围中的元素会被排序(这点和归并排序的过程一样)。

那么函数又是如何快速统计逆序对的呢?大体过程如下:

  • 递归调用,拿到左子数组和右子数组的逆序对(此时,左子数组和右子数组也都排序完成了)
  • 指针 i 和 j 分别指向左子数组和右子数组的最右侧,此时会有 2 种情况:
    • arr[i] > arr[j]:那么说明arr[i]大于右子数组中所有元素,逆序对添加j - start - length,向左边移动指针 i
    • arr[i] <= arr[j]: 对arr[i]来说,不存在逆序对,向左边移动指针 j
  • i 和 j 遍历完各自数组后,最后返回逆序对之和就可

代码实现如下:

// ac地址:https://leetcode-cn.com/problems/shu-zu-zhong-de-ni-xu-dui-lcof/// 原文地址:https://xxoo521.com/2020-03-12-reverse-pairs//** * @param {number[]} nums * @return {number} */var reversePairs = function(nums) { return findInversePairNum(nums, 0, nums.length - 1);};/** * @param {number[]} arr * @param {number} start * @param {number} end */function findInversePairNum(arr, start, end) { if (start >= end) return 0; const copy = new Array(end - start + 1); const length = Math.floor((end - start) / 2); // 左数组长度 const leftNum = findInversePairNum(arr, start, start + length); const rightNum = findInversePairNum(arr, start + length + 1, end); let i = start + length; let j = end; let copyIndex = end - start; let num = 0; while (i >= start && j >= start + length + 1) { if (arr[i] > arr[j]) { num += j - start - length; copy[copyIndex--] = arr[i--]; } else { copy[copyIndex--] = arr[j--]; } } while (i >= start) { copy[copyIndex--] = arr[i--]; } while (j >= start + length + 1) { copy[copyIndex--] = arr[j--]; } for (let k = start; k <= end; ++k) { arr[k] = copy[k - start]; } return num + leftNum + rightNum;}

时间复杂度是O(NlogN),空间复杂度是O(N)。假如还是觉得不好了解,可以以数组 7、5、6、4 为例,按照前面过程,手动计算一下。

更多资料

整理不易,若对您有帮助,请给个「关注+点赞」,您的支持是我升级的动力 ??

  • ??Blog:剑指 Offer 题解 + JS 代码
  • ??Github : dongyuanxin/blog
  • ?? 公众号:心谭博客

推荐阅读
  • 目录实现效果:实现环境实现方法一:基本思路主要代码JavaScript代码总结方法二主要代码总结方法三基本思路主要代码JavaScriptHTML总结实 ... [详细]
  • 闭包一直是Java社区中争论不断的话题,很多语言都支持闭包这个语言特性,闭包定义了一个依赖于外部环境的自由变量的函数,这个函数能够访问外部环境的变量。本文以JavaScript的一个闭包为例,介绍了闭包的定义和特性。 ... [详细]
  • 使用nodejs爬取b站番剧数据,计算最佳追番推荐
    本文介绍了如何使用nodejs爬取b站番剧数据,并通过计算得出最佳追番推荐。通过调用相关接口获取番剧数据和评分数据,以及使用相应的算法进行计算。该方法可以帮助用户找到适合自己的番剧进行观看。 ... [详细]
  • 本文介绍了闭包的定义和运转机制,重点解释了闭包如何能够接触外部函数的作用域中的变量。通过词法作用域的查找规则,闭包可以访问外部函数的作用域。同时还提到了闭包的作用和影响。 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • javascript  – 概述在Firefox上无法正常工作
    我试图提出一些自定义大纲,以达到一些Web可访问性建议.但我不能用Firefox制作.这就是它在Chrome上的外观:而那个图标实际上是一个锚点.在Firefox上,它只概述了整个 ... [详细]
  • 使用在线工具jsonschema2pojo根据json生成java对象
    本文介绍了使用在线工具jsonschema2pojo根据json生成java对象的方法。通过该工具,用户只需将json字符串复制到输入框中,即可自动将其转换成java对象。该工具还能解析列表式的json数据,并将嵌套在内层的对象也解析出来。本文以请求github的api为例,展示了使用该工具的步骤和效果。 ... [详细]
  • Voicewo在线语音识别转换jQuery插件的特点和示例
    本文介绍了一款名为Voicewo的在线语音识别转换jQuery插件,该插件具有快速、架构、风格、扩展和兼容等特点,适合在互联网应用中使用。同时还提供了一个快速示例供开发人员参考。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • PHP图片截取方法及应用实例
    本文介绍了使用PHP动态切割JPEG图片的方法,并提供了应用实例,包括截取视频图、提取文章内容中的图片地址、裁切图片等问题。详细介绍了相关的PHP函数和参数的使用,以及图片切割的具体步骤。同时,还提供了一些注意事项和优化建议。通过本文的学习,读者可以掌握PHP图片截取的技巧,实现自己的需求。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 安卓select模态框样式改变_微软Office风格的多端(Web、安卓、iOS)组件库——Fabric UI...
    介绍FabricUI是微软开源的一套Office风格的多端组件库,共有三套针对性的组件,分别适用于web、android以及iOS,Fab ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • Html5-Canvas实现简易的抽奖转盘效果
    本文介绍了如何使用Html5和Canvas标签来实现简易的抽奖转盘效果,同时使用了jQueryRotate.js旋转插件。文章中给出了主要的html和css代码,并展示了实现的基本效果。 ... [详细]
  • 本文总结了在开发中使用gulp时的一些技巧,包括如何使用gulp.dest自动创建目录、如何使用gulp.src复制具名路径的文件以及保留文件夹路径的方法等。同时介绍了使用base选项和通配符来保留文件夹路径的技巧,并提到了解决带文件夹的复制问题的方法,即使用gulp-flatten插件。 ... [详细]
author-avatar
yixianliu
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有