力扣2080. 区间内查询数字的频率


这一题的大意是让我们寻找在一个数组中的某一子数组中某一个元素出现的次数。

找一个元素在某一数组中出现的次数,我们通常会想到的是遍历一遍数组用哈希表来统计该元素出现的次数,但问题是这道题中会有最多10^5次询问,而数组的长度最多为 10 ^5,暴力遍历肯定是超时的,这时候我们可以采用二分的形式,通过找该元素最早出现的位置和该元素最晚出现的位置,两者相减就是出现的次数,时间复杂度O(nlogn)可以接受

但是要注意的是虽然使用二分可以下降到O(logn)但是二分需要在单调区间中才能使用,而题目中给出的序列是无序的,我们如果在每一次进行二分前进行一个排序的话,那么时间复杂度会超时,O(n)*(nlogn)(排序的时间复杂度 * n次询问)。

我之前写的就是排序+二分的做法超时了:

cpp 复制代码
class RangeFreqQuery {
public:
    vector<int> a;
    RangeFreqQuery(vector<int>& arr) {
        for(int i=0;i<arr.size();i++)
        {
            a.push_back(arr[i]);
        }
    }
    int query(int left, int right, int value) {
            //排序
            vector<int> temp=a;
            sort(temp.begin()+left,temp.begin()+right+1);
            int l=left;
            int r=right;
            //找第一个大于等于value的位置
            //找第一个小于等于value的位置
            //如果数组中有value 那么
            // 那么这两个位置都一定在value上
            int lx=INT_MAX;
            while(l<=r)
            {
              int mid=(l+r)/2;
              if(temp[mid]==value)
              {
                 r=mid-1;
                 lx=min(lx,mid);
              }
              else if(temp[mid]>value)
              {
                r=mid-1;
              }
              else{
                l=mid+1;
              }
            } 
            //找到最左端的lx value
             int rx=-1;
             l=left;
             r=right;
             while(l<=r)
             {
                int mid=(l+r)/2;
                if(temp[mid]==value)
                {
                    l=mid+1;
                    rx=max(rx,mid);
                }
                else if(temp[mid]>value){
                    r=mid-1;
                }
                else{
                    l=mid+1;
                }
             }
             //这样就找到了lx和lr的位置了
             if(lx==INT_MAX||rx==-1)
             {
                return 0;
             }
             else 
             {
                return rx-lx+1;
             }
    }
};

/**
 * Your RangeFreqQuery object will be instantiated and called as such:
 * RangeFreqQuery* obj = new RangeFreqQuery(arr);
 * int param_1 = obj->query(left,right,value);
 */

实际应该用哈希+二分,我们在询问之前,就用哈希表统计出每一个值在数组中可能出现的位置,然后我们用一个二分来看可能存在的位置是否在询问的区间内,分别找到出现在区间内的最小位置和最大位置,二者之差即为出现的频率次数。

完整的代码如下:

cpp 复制代码
class RangeFreqQuery {
public:
    vector<int> a;
    unordered_map<int,vector<int> > mp;
    RangeFreqQuery(vector<int>& arr) {
        for(int i=0;i<arr.size();i++)
        {
            a.push_back(arr[i]);
            mp[arr[i]].push_back(i);
        }
    }
    int query(int left, int right, int value) {
            //排序
            vector<int>& temp=mp[value];
            if(mp[value].size()==0)
            {
            return 0;     
            }
            int l=0;
            int r=temp.size()-1;
            //找第一个大于等于value的位置
            //找第一个小于等于value的位置
            //如果数组中有value 那么
            // 那么这两个位置都一定在value上
            int lx=INT_MAX;
            while(l<=r)
            {
              int mid=(l+r)/2;
              if(temp[mid]>=left&&temp[mid]<=right)
              {
                 r=mid-1;
                 lx=min(lx,mid);
              }
              else if(temp[mid]>right)
              {
                r=mid-1;
              }
              else{
                l=mid+1;
              }
            } 
            //找到最左端的lx value
             int rx=-1;
             l=0;
             r=temp.size()-1;
             while(l<=r)
             {
                int mid=(l+r)/2;
                if(temp[mid]>=left&&temp[mid]<=right)
                {
                    l=mid+1;
                    rx=max(rx,mid);
                }
                else if(temp[mid]>right){
                    r=mid-1;
                }
                else{
                    l=mid+1;
                }
             }
             //这样就找到了lx和lr的位置了
             if(lx==INT_MAX||rx==-1)
             {
                return 0;
             }
             else 
             {
                return rx-lx+1;
             }
    }
};

/**
 * Your RangeFreqQuery object will be instantiated and called as such:
 * RangeFreqQuery* obj = new RangeFreqQuery(arr);
 * int param_1 = obj->query(left,right,value);
 */

需要注意的是,在获取哈希表中的某一个值对应的出现位置的数组时,我们要用引用来获取,否则会超时

cpp 复制代码
vector<int>& temp=mp[value];

最终的时间复杂度是O(nlogn)

相关推荐
电鱼智能的电小鱼3 小时前
基于电鱼 AI 工控机的智慧工地视频智能分析方案——边缘端AI检测,实现无人值守下的实时安全预警
网络·人工智能·嵌入式硬件·算法·安全·音视频
孫治AllenSun3 小时前
【算法】图相关算法和递归
windows·python·算法
格图素书4 小时前
数学建模算法案例精讲500篇-【数学建模】DBSCAN聚类算法
算法·数据挖掘·聚类
DashVector5 小时前
向量检索服务 DashVector产品计费
数据库·数据仓库·人工智能·算法·向量检索
AI纪元故事会5 小时前
【计算机视觉目标检测算法对比:R-CNN、YOLO与SSD全面解析】
人工智能·算法·目标检测·计算机视觉
夏鹏今天学习了吗5 小时前
【LeetCode热题100(59/100)】分割回文串
算法·leetcode·深度优先
卡提西亚6 小时前
C++笔记-10-循环语句
c++·笔记·算法
还是码字踏实6 小时前
基础数据结构之数组的双指针技巧之对撞指针(两端向中间):三数之和(LeetCode 15 中等题)
数据结构·算法·leetcode·双指针·对撞指针
Coovally AI模型快速验证8 小时前
当视觉语言模型接收到相互矛盾的信息时,它会相信哪个信号?
人工智能·深度学习·算法·机器学习·目标跟踪·语言模型
电院工程师8 小时前
SIMON64/128算法Verilog流水线实现(附Python实现)
python·嵌入式硬件·算法·密码学