LeetCode 2080. 区间内查询数字的频率(哈希+二分查找)

简介: LeetCode 2080. 区间内查询数字的频率(哈希+二分查找)

文章目录


1. 题目

2. 解题

2.1 超时解

2.2 优化


1. 题目


请你设计一个数据结构,它能求出给定子数组内一个给定值的 频率 。


子数组中一个值的 频率 指的是这个子数组中这个值的出现次数。


请你实现 RangeFreqQuery 类:


RangeFreqQuery(int[] arr) 用下标从 0 开始的整数数组 arr 构造一个类的实例。

int query(int left, int right, int value) 返回子数组 arr[left...right] 中 value 的 频率 。

一个 子数组 指的是数组中一段连续的元素。arr[left...right] 指的是 nums 中包含下标 left 和 right 在内 的中间一段连续元素。

示例 1:
输入:
["RangeFreqQuery", "query", "query"]
[[[12, 33, 4, 56, 22, 2, 34, 33, 22, 12, 34, 56]], [1, 2, 4], [0, 11, 33]]
输出:
[null, 1, 2]
解释:
RangeFreqQuery rangeFreqQuery = new RangeFreqQuery([12, 33, 4, 56, 22, 2, 34, 33, 22, 12, 34, 56]);
rangeFreqQuery.query(1, 2, 4); // 返回 1 。4 在子数组 [33, 4] 中出现 1 次。
rangeFreqQuery.query(0, 11, 33); // 返回 2 。33 在整个子数组中出现 2 次。
提示:
1 <= arr.length <= 10^5
1 <= arr[i], value <= 10^4
0 <= left <= right < arr.length
调用 query 不超过 10^5 次。



2. 解题


2.1 超时解


  • 存储每个位置的每个数的出现次数的前缀和
class RangeFreqQuery {
    vector<unordered_map<int,int>> presum;
public:
    RangeFreqQuery(vector<int>& arr) {
        presum.reserve(arr.size());
        presum.emplace_back(unordered_map<int,int>());
        presum[0][arr[0]] = 1;
        for(int i = 1; i < arr.size(); ++i)
        {
            presum.emplace_back(presum.back());
            if(presum[i-1].find(arr[i]) == presum[i-1].end())
                presum[i][arr[i]] = 1;
            else
                presum[i][arr[i]] = presum[i-1][arr[i]]+1;
        }
    }
    int query(int left, int right, int value) {
        if(presum[right].find(value) == presum[right].end())
            return 0;
        if(left == 0 || presum[left-1].find(value) == presum[left-1].end())
            return presum[right][value];
        return presum[right][value] - presum[left-1][value];
    }
};



2.2 优化


  • 记录同一种数字的所有的位置,进行二分查找 left 和 right 的 位置
class RangeFreqQuery {
    unordered_map<int, vector<int>> pos; // 数字出现的位置
public:
    RangeFreqQuery(vector<int>& arr) {
        for(int i = 0; i < arr.size(); ++i)
        {
            pos[arr[i]].push_back(i);
        }
    }
    int query(int left, int right, int value) {
        if(pos.find(value) == pos.end()) return 0;
        auto it1 = lower_bound(pos[value].begin(), pos[value].end(), left);
        auto it2 = upper_bound(pos[value].begin(), pos[value].end(), right);
        return it2-it1;
    }
};

524 ms 245.6 MB C++

相关文章
|
2月前
|
算法
Leetcode第57题(插入区间)
LeetCode第57题“插入区间”的解题方法,包括题目描述、示例、算法思路和代码实现,旨在解决将新区间插入有序且不重叠的区间列表中,并合并重叠区间的问题。
22 0
Leetcode第57题(插入区间)
|
2月前
【LeetCode 01】二分查找总结
【LeetCode 01】二分查找总结
17 0
|
4月前
|
算法
LeetCode第57题插入区间
LeetCode第57题"插入区间"的解题方法,利用原区间集有序的特性,通过三步插入操作,有效实现了新区间的插入和重叠区间的合并。
LeetCode第57题插入区间
|
4月前
|
Python
【Leetcode刷题Python】704. 二分查找
解决LeetCode "二分查找" 问题的Python实现代码。
20 0
|
4月前
|
算法 索引 Python
【Leetcode刷题Python】34. 在排序数组中查找元素的第一个和最后一个位置(二分查找)
解决LeetCode "在排序数组中查找元素的第一个和最后一个位置" 问题的方法。第一种方法是使用两次二分查找,首先找到目标值的最左边界,然后找到最右边界。第二种方法是利用Python的list.index()方法,先正序找到起始位置,再逆序找到结束位置,并给出了两种方法的Python实现代码。
65 0
|
6月前
|
存储 算法 测试技术
力扣经典150题第四十七题:汇总区间
力扣经典150题第四十七题:汇总区间
49 1
|
6月前
|
索引
【LeetCode刷题】二分查找:山脉数组的峰顶索引、寻找峰值
【LeetCode刷题】二分查找:山脉数组的峰顶索引、寻找峰值
|
6月前
|
算法 数据挖掘 开发者
LeetCode题目55:跳跃游戏【python5种算法贪心/回溯/动态规划/优化贪心/索引哈希映射 详解】
LeetCode题目55:跳跃游戏【python5种算法贪心/回溯/动态规划/优化贪心/索引哈希映射 详解】
|
6月前
【LeetCode刷题】专题三:二分查找模板
【LeetCode刷题】专题三:二分查找模板
【LeetCode刷题】专题三:二分查找模板
|
6月前
|
存储 SQL 算法
高效日程管理:利用区间合并算法优化活动安排【python LeetCode57】
高效日程管理:利用区间合并算法优化活动安排【python LeetCode57】