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

题目链接

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();
};
相关推荐
裕波1 小时前
AI 正在重写应用开发。Vue 与 Vite,给出新的答案。
javascript·vue.js
kyriewen2 小时前
折腾了半年 AI 编程工作流,最后发现效率瓶颈是桌上那块屏幕
前端·javascript·ai编程
张元清5 小时前
React useDebounce Hook:给状态和回调做防抖(2026)
javascript·react.js
Cobyte6 小时前
21.Vue Vapor 组件的实现原理
前端·javascript·vue.js
铁皮饭盒6 小时前
Rust版Bun1.4之前, 盘点Bun1.3新特性
前端·javascript·后端
晓得迷路了6 小时前
栗子前端技术周刊第 135 期 - Vite 8.1、Rspack 2.1、Babel 8.0...
前端·javascript·vite
To_OC15 小时前
LC 207 课程表:刚学图论那会儿,我连这是拓扑排序都没看出来
javascript·算法·leetcode
To_OC15 小时前
LC 208 实现 Trie 前缀树:曾被名字劝退,写完发现是送分题
javascript·算法·leetcode
天渺工作室16 小时前
实现一个adblock/adblock plus等浏览器广告拦截器检测插件
前端·javascript
BadBadBad__AK17 小时前
线段树维护区间 k 次方和
c++·数学·算法·stl