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

题目链接

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

  • 树状数组:

树状数组详解

思路:

先排序然后构建一个树状数组,从后往前开始计算遍历构建一个树状数组,根据当前的 numsi 缓存下标到 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();
};
相关推荐
梦想的颜色8 小时前
TypeScript 完全指南(下):从类型体操到生产级配置
前端·javascript·typescript
8Qi89 小时前
LeetCode 75:颜色分类(荷兰国旗问题)—— Java 题解 ✅
java·算法·leetcode·指针·排序
888CC++10 小时前
如何在 C 语言中进行程序调试?
前端·javascript·算法
pluviophile_s12 小时前
数据结构:第2讲:线性表
数据结构·笔记
(●—●)橘子……12 小时前
力扣第503场周赛练习理解
python·学习·算法·leetcode·职场和发展·周赛
kyriewen13 小时前
我招了一个“Prompt工程师”来写前端,结果项目差点崩了
前端·javascript·面试
小新11013 小时前
从零开始 Vue.js
前端·javascript·vue.js
Delicate13 小时前
JavaScript的“变脸”艺术:类型转换戏法大揭秘
javascript
明志数科13 小时前
4D时序标注技术详解:让机器人理解连续动作的数据基础
java·算法·机器人