文章目录

  • 1 题目理解
  • 2 暴力解法
  • 3 分治法

1 题目理解

输入:int[] nums
输出:计数的数组int[] counts
规则:counts[i]表示nums中下标大于i,值小于nums[i]的个数
Example 1:
Input: nums = [5,2,6,1]
Output: [2,1,1,0]
Explanation:
To the right of 5 there are 2 smaller elements (2 and 1).
To the right of 2 there is only 1 smaller element (1).
To the right of 6 there is 1 smaller element (1).
To the right of 1 there is 0 smaller element.

2 暴力解法

对于处理每个元素nums[i],依次数一数从i+1到n,小于nums[i]的数有多少个。

class Solution {public List<Integer> countSmaller(int[] nums) {List<Integer> list = new ArrayList<Integer>();for(int i=0;i<nums.length;i++){int c = 0;for(int j=i+1;j<nums.length;j++){if(nums[j]<nums[i]){c++;}}list.add(c);}return list;}
}

时间复杂度O(n2)O(n^2)O(n2)

3 分治法

以下内容来自力扣官网。
数一数在右侧,比自己小的元素个数,这实际上就是在求逆序对。在学习算法过程中遇到过,利用归并排序计算逆序对个数。

我们还是要在「归并排序」的「并」中做文章。我们通过一个实例来看看。假设我们有两个已排序的序列等待合并,分别是 L = { 8, 12, 16, 22, 100 } 和 R = { 7, 26, 55, 64, 91 }。一开始我们用指针 lPtr = 0 指向 L 的头部,rPtr = 0 指向 R 的头部。记已经合并好的部分为 M。

我们发现 lPtr 指向的元素大于 rPtr 指向的元素,于是把 rPtr 指向的元素放入答案,并把 rPtr 后移一位。

接着我们继续合并:

此时 lPtr 比 rPtr 小,把 lPtr 对应的数加入答案。如果我们要统计 8 的右边比 8 小的元素,这里 7 对它做了一次贡献。如果待合并的序列 L = { 8, 12, 16, 22, 100 },R = { 7, 7, 7, 26, 55, 64, 91},那么一定有一个时刻,lPtr 和 rPtr 分别指向这些对应的位置:

下一步我们就是把 8 加入 M 中,此时三个 7 对 8 的右边比 8 小的元素的贡献为 3。以此类推,我们可以一边合并一边计算 R 的头部到 rPtr 前一个数字对当前 lPtr 指向的数字的贡献。

我们发现用这种「算贡献」的思想在合并的过程中计算逆序对的数量的时候,只在 lPtr 右移的时候计算,是基于这样的事实:当前 lPtr 指向的数字比 rPtr 小,但是比 R 中 [0 … rPtr - 1] 的其他数字大,[0 … rPtr - 1] 的数字是在 lPtr 右边但是比 lPtr 对应数小的数字,贡献为这些数字的个数。

但是我们又遇到了新的问题,在「并」的过程中 88 的位置一直在发生改变,我们应该把计算的贡献保存到哪里呢?这个时候我们引入一个新的数组,来记录每个数字对应的原数组中的下标,例如:

排序的时候原数组和这个下标数组同时变化,则排序后我们得到这样的两个数组:


我们用一个数组 ans 来记录贡献。我们对某个元素计算贡献的时候,如果它对应的下标为 p,我们只需要在 ans[p] 上加上贡献即可。

class Solution {private int[] index;private int[] temp;private int[] tempIndex;private int[] ans;private Map<Integer,Integer> valueIndexMap;public List<Integer> countSmaller(int[] nums) {this.index = new int[nums.length];this.temp = new int[nums.length];this.tempIndex = new int[nums.length];this.ans = new int[nums.length];valueIndexMap = new HashMap<Integer,Integer>();for(int i=0;i<nums.length;i++){valueIndexMap.put(nums[i],i);}for (int i = 0; i < nums.length; ++i) {index[i] = i;}int l = 0, r = nums.length - 1;mergeSort(nums, l, r);List<Integer> list = new ArrayList<Integer>();for (int num : ans) {list.add(num);}return list;}public void mergeSort(int[] a, int l, int r) {if (l >= r) {return;}int mid = (l + r) >> 1;mergeSort(a, l, mid);mergeSort(a, mid + 1, r);merge(a, l, mid, r);}public void merge(int[] a, int l, int mid, int r) {int i = l, j = mid + 1, p = l;while (i <= mid && j <= r) {if (a[i] <= a[j]) {temp[p] = a[i];tempIndex[p] = index[i];ans[valueIndexMap.get(a[i])] += (j - mid - 1);++i;++p;} else {temp[p] = a[j];tempIndex[p] = index[j];++j;++p;}}while (i <= mid)  {temp[p] = a[i];tempIndex[p] = index[i];ans[valueIndexMap.get(a[i])] += (j - mid - 1);++i;++p;}while (j <= r) {temp[p] = a[j];tempIndex[p] = index[j];++j;++p;}for (int k = l; k <= r; ++k) {index[k] = tempIndex[k];a[k] = temp[k];}}
}

时间复杂度O(nlogn)O(nlogn)O(nlogn)

315. Count of Smaller Numbers After Self相关推荐

  1. 315. Count of Smaller Numbers After Self 计算右侧小于当前元素的个数

    Title 给定一个整数数组 nums,按要求返回一个新数组 counts.数组 counts 有该性质: counts[i] 的值是 nums[i] 右侧小于 nums[i] 的元素的数量. 示例: ...

  2. C++版 - 剑指Offer 面试题36:数组中的逆序对及其变形(Leetcode 315. Count of Smaller Numbers After Self)题解

    剑指Offer 面试题36:数组中的逆序对 题目:在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对.输入一个数组,求出这个数组中的逆序对的总数. 例如, 在数组{7,5, ...

  3. LeetCode 315. 计算右侧小于当前元素的个数(Count of Smaller Numbers After Self)

    题目描述: 给定一个整数数组 nums,按要求返回一个新数组 counts.数组 counts 有该性质: counts[i] 的值是  nums[i] 右侧小于 nums[i] 的元素的数量. 示例 ...

  4. 递归/归并:count of smaller numbers求逆序数

    已知数组nums,求新数组count,count[i]代表了在nums[i]右侧且比 nums[i]小的元素个数. 例如: nums = [5, 2, 6, 1], count = [2, 1, 1, ...

  5. 第七章第三题(计算数字的出现次数)(Count occurrence of numbers) - 编程练习题答案

    编写程序,读取在1到100 之间的整数,然后计算每个数出现的次数.假定输入是以0 结束的. 下面是这个程序的一个运行示例: Write a program that reads the integer ...

  6. Merge Sort及其对一类问题的应用

    1.归并排序 O(nlogn) stable #include <iostream> #include <vector> using namespace std;void me ...

  7. 数据结构与算法总结(完结)

    极客时间算法学习之后开始跟着花花酱刷题.大概从4月份开始的.从今天开始(2020-8-24)开始做总结,复习一下已经刷过的题目.到目前为止leetcode刷题323道. 2020/8/24 完成题目整 ...

  8. 1218. 最长定差子序列

    文章目录 1 题目理解 2 开始思考 1 题目理解 给你一个整数数组 arr 和一个整数 difference,请你找出并返回 arr 中最长等差子序列的长度,该子序列中相邻元素之间的差等于 diff ...

  9. LeetCode 406. Queue Reconstruction by Height

    原题链接在这里:https://leetcode.com/problems/queue-reconstruction-by-height/description/ 题目: Suppose you ha ...

最新文章

  1. 华为odc是什么意思_三星S20 FE官宣;华为Mate 40系列中国独占发售
  2. windows批处理使用记录
  3. 面试必问之【数组】篇
  4. eclipse 设置豆沙绿保护色,保护眼睛
  5. Lucene3.5自学4--建索引相关知识总结
  6. Navicat远程服务器2013-Lost connection to MYSQL server at 'reading for initial communication packet' 公钥
  7. c++反汇编代码分析--偷调函数
  8. 别人连不上_手机能连上wifi,笔记本电脑确连不上,显示无internet,安全,该怎么办?...
  9. R750 H750 H755 阵列卡磁盘次序
  10. Multispectral Deep Neural Networks for Pedestrian Detection(BMVC 2016)论文解读
  11. 华为笔试:拼音与英文互相转换python
  12. 计算机电子钢琴,电脑钢琴调律
  13. 能把晦涩难懂的研究工作讲清楚,Distill就奖你10000美刀
  14. heidisql修改mysql密码_读取HeidiSQL 配置文件中的密码
  15. 1447_TC275 DataSheet阅读笔记8_电气特性
  16. 互联网产品分析-基础
  17. IC公司认可的数字IC_FPGA设计课程
  18. python写软件实例-python写一个随机点名软件的实例
  19. 现代操作系统及进程管理(思维导图)
  20. 【一起学UniGUI】--UniGUI的界面与程序架构(4)

热门文章

  1. [Tomcat报错]SEVERE: Error listenerStart
  2. mysql区分大小写搜索
  3. 再看数据库——(2)视图
  4. 动态绑定 datagridview
  5. 三全食品:信息化建设狂飙突进的六年
  6. 使用默认Model Binding支持集合类
  7. 计算机专硕专业课单科分数线,计算机考研|这两所自划线,单科没过线也能复试?...
  8. php 识别语种,HYPHP增加多国语言支持 PHP通过用户浏览器判断来源国家方案
  9. iframe带了token不显示_不就是登录吗,能有多复杂?sa-token带你轻松搞定多地登陆、单地登录、同端互斥登录...
  10. Android修改项目包名