2022/11/09 算法练习 力扣 2080. 区间内查询数字的频率

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 <= arr.length <= 105
  • 1 <= arr[i], value <= 104
  • 0 <= left <= right < arr.length
  • 调用 query 不超过 105 次。

示例 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 次。

2、完整代码

class RangeFreqQuery {
    Map<Integer,List<Integer>> map;
    public RangeFreqQuery(int[] arr) {
        map = new HashMap<>();
        for(int i=0;i<arr.length;i++){
            List<Integer> list = map.getOrDefault(arr[i],new ArrayList<>());
            list.add(i);
            map.put(arr[i],list);
        }
    }
    public int query(int left, int right, int value) {
       List<Integer> list = map.get(value);
       if(list == null){
           return 0;
       }else{
           int l =LeftBinarySearch(left,list);
           int r =RightBinarySearch(right,list);
           return r-l;
        }   
    }
    public int LeftBinarySearch(int left,List<Integer> list){
        int l = 0;
        int r = list.size();
        while(l < r){
            int mid = (r+l)/2;
            if(left <= list.get(mid))   r = mid;
            else   l = mid + 1;
        }
        return l;
    }
     public int RightBinarySearch(int right,List<Integer> list){
        int l = 0;
        int r = list.size();
        while(l < r){
            int mid = (r+l)/2;
            if(right >= list.get(mid))   l = mid+1;
            else   r = mid;
        }
        return l;
    }
}

3、分析

 

class RangeFreqQuery{
    private int[] a;
    public RangeFreqQuery(int[] arr) {
        a = arr;
    }
    public int query(int left, int right, int value) {
        int sum=0;
        for(int i=left;i<=right;i++){
            if(value == a[i])  sum++;
        }
        return sum;
    }
}
  • 一开始看到题目我还说怎么这个题目会是中等难度,然后就直接拿到数组开始循环累计出现次数,失败之后发现这道题目的难度在于他对时间复杂度有一定的要求。怪不得要让我们实现类。

  • 那么就要给定的类中将给定的数组安排好,转换成另一种形式,方便query计算,以达到降低时间复杂度的需求。
    • 首先想到哈希表,用Map直接记录每一个元素的对应出现的位置
    • 然后在Map中的List集合中拿到在left和right之间的下标,计算下标数量
  • 本来想把二分查找合并,结果发现合并之后因为要判断是找左边还是右边,所以时间复杂度变高,所以就放弃了
public int BinarySearch(int aim,List<Integer> list,int flag){
        int l = 0;
        int r = list.size();
        if(flag==0){   //找小的,向右边靠
            while(l < r){
            int mid = (r+l)/2;
            if(aim <= list.get(mid))   r = mid;
            else   l = mid + 1;
            }
        }   
        if(flag==1){     //找大的,向左边靠
             while(l < r){
            int mid = (r+l)/2;
            if(aim >= list.get(mid))   l = mid+1;
            else   r = mid;
             }
        }
        return l;
    }

  • 去看了时间复杂度最少的代码,心情稍稍有点复杂...
class RangeFreqQuery {

    private int[] arr;
    private int[] freq;

    public RangeFreqQuery(int[] arr) {
        int n = arr.length;
        this.arr = arr;
        this.freq = new int[n];
        int[] cnt = new int[10001];
        for (int i = 0; i < n; i++) {
            freq[i] = cnt[arr[i]]++;
        }
    }
    public int query(int left, int right, int value) {
        int l = right, r = left;
        for (int i = left; i <= right; i++) {
            if (arr[i] == value) {
                l = i;
                break;
            }
        }
        for (int i = right; i >= l; i--) {
            if (arr[i] == value) {
                r = i;
                break;
            }
        }
        return arr[l] != value ? 0 : freq[r] - freq[l] + 1;
    }

  • 7
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值