计算右侧小于当前元素的个数 题解

题目链接

315. 计算右侧小于当前元素的个数 - 力扣(LeetCode)

题解

前置概念:

  • in 和 of 区别

in 操作符:遍历或者检查对象的键名,遍历的是可枚举属性名,不适合遍历数组,会遍历原型链上的属性

of 操作符:遍历可迭代对象的键值(数组,字符串,Map,Set 等,不能遍历普通对象,除非定义了 Symbol.iterotor)不会遍历原型链

  • size 和 length 区别

length 适合用于类数组 (数组,字符串),表示索引的个数

size 用于集合类型,表示键值对/元素的个数

  • lowBit 函数

作用是 只保留一个整数的二进制中最右边的1,并将其余位置全部为0

利用了计算机中负数以补码的形式存储的特性

为什么 x & (-x) 能留下最右边的 1?

  1. 对于 x 的最右边的 1,设它在第 k 位

  2. 这个 1 右边的所有位都是 0

  3. 当对 x 取反时,第 k 位变成 0,右边的 0 都变成 1

  4. 再加 1 时,进位会把第 k 位重新变成 1,而右边又变回 0

  5. 这样,-x 的第 k 位也是 1,且第 k 位右边的位都是 0

  6. 所以 x & (-x) 的结果中,只有第 k 位是 1

  • 树状数组:

树状数组详解

思路:

先排序然后构建一个树状数组,从后往前开始计算遍历构建一个树状数组,根据当前的 nums[i] 缓存下标到 map 里面(没用二分查找是因为时间超限,使用这个是以空间换时间,事件复杂度从logn 到 1),最后输出倒序的结果即可

代码

javascript 复制代码
var countSmaller = function(nums) {
    let c = [];
    let a = [];
    let valueToIndex = new Map();

    function init(length) {
        return new Array(length + 5).fill(0);
    }

    function lowBit(x) {
        return x & (-x);
    }

    function update(pos) {
        while (pos < c.length) {
            c[pos]++;
            pos += lowBit(pos);
        }
    }

    function query(pos) {
        let resNum = 0;
        while (pos > 0) {
            resNum += c[pos];
            pos -= lowBit(pos);
        }
        return resNum;
    }

    function discretization(nums) {
        // 去重并排序
        let set = new Set(nums);
        a = Array.from(set).sort((x, y) => x - y);
        
        // 建立映射(索引从1开始)
        a.forEach((val, idx) => {
            valueToIndex.set(val, idx + 1);
        });
    }

    function getRes() {
        let res = [];
        discretization(nums);
        c = init(nums.length);
        
        for (let i = nums.length - 1; i >= 0; i--) {
            let id = valueToIndex.get(nums[i]); // O(1) 获取
            res.push(query(id - 1)); // 查询比它小的个数
            update(id); // 把自己加进去
        }
        
        return res.reverse();
    }

    return getRes();
};
相关推荐
滴滴答滴答答1 小时前
机考刷题之 12 LeetCode 684 冗余的边
算法·leetcode·职场和发展
天天向上10241 小时前
vue 大屏适配的一种实现思路
前端·javascript·vue.js
SuperEugene1 小时前
Vue/Vite 多环境配置实战:dev、test、prod 差异区分与避坑指南|Vue 工程化篇
前端·javascript·vue.js
美式请加冰2 小时前
前缀数组的介绍和使用
数据结构·c++·算法
GawynKing2 小时前
图论2 图的数据结构表示
数据结构·图论
IronMurphy2 小时前
【算法十九】33. 搜索旋转排序数组 74. 搜索二维矩阵
线性代数·算法·矩阵
lichenyang4532 小时前
React 性能优化组件设计模式与通信
前端·javascript·设计模式
Srend6662 小时前
【图论】最短路问题
算法·图论
云泽8082 小时前
蓝桥杯算法精讲:二分算法之二分查找深度剖析
算法·职场和发展·蓝桥杯