首页 > 其他分享 >795. 区间子数组个数

795. 区间子数组个数

时间:2023-04-04 23:56:26浏览次数:34  
标签:795 right 数组 int 个数 last1 端点 last2

题目描述

给一个数组,再给一个值的范围[l, r],
问最大值在[l, r]之间的子数组有多少个?

f1-双指针

基本分析

  1. 如果枚举子数组的右端点i,会有几种情况?(1)arr[i] > right; (left <= arr[i] <= right; (3)arr[i] < left
  2. 假如枚举到右端点i,左端点怎么考虑?(1)的情况,这个子数组不满足,可以跳过;(2)的情况i可以是一个最近的左端点,只需要找到最远的左端点i就行。分两种情况,前面没有>right的索引,i就是0;前面有>right的索引,i就是索引+1;(3)的情况,自己不能当左端点,除了找最远的左端点以外,还要找最近的左端点i’,可选的左端点个数是i' - i
  3. 怎么实现以上逻辑?用两个指针last1表示最近的左端点位置;last2表示最近的>right的位置。对枚举到的每个值,当last1存在的时候,表示有左端点存在,就可以累加结果

代码

class Solution:
    def numSubarrayBoundedMax(self, nums: List[int], left: int, right: int) -> int:
        last1 = last2 = -1
        ans = 0

        for i, x in enumerate(nums):
            if left <= x <= right:
                last1 = i
            elif x > right:
                last2 = i
                last1 = -1
            if last1 != -1:
                ans += last1 - last2
        
        return ans

总结

  1. 枚举右端点,右端点可以保证子数组不同
  2. 个数取决于可选的左端点范围,这里用last1和last2来维护
f2-计数

基本分析

  1. 怎么用计数的角度考虑?(1)找<= right的区间个数(2)找<=left-1的区间个数;(3)求差
  2. 怎么统计<=x的区间个数?如果长度是l,个数就是1+2+3+...+l,碰到不满足的重新计数。

代码

class Solution:
    def numSubarrayBoundedMax(self, nums: List[int], left: int, right: int) -> int:

        def cal(x):
            ans = 0
            cnt = 0
            for num in nums:
                if num <= x:
                    cnt += 1
                else:
                    cnt = 0
                ans += cnt
            return ans
        
        return cal(right) - cal(left - 1)

总结

  1. 利用题目的要求,压缩每个数字的含义,使其变为简单的0、1和2
  2. 利用容斥的思想,使用计数解决
f3-单调栈
待补充

基本分析

  1. xxx

代码


总结

  1. xxx

标签:795,right,数组,int,个数,last1,端点,last2
From: https://www.cnblogs.com/zk-icewall/p/17288307.html

相关文章

  • 3377. 约数的个数(约数个数)
    https://www.acwing.com/problem/content/3380/这题和第11届蓝桥杯B组国赛题类似数论知识,就是分解质因数,把质数的指数加1即可需要注意的是,本题应该是不能用数组模拟的,空间太少了可以用unordered_map存储#include<iostream>#include<cstdio>#include<cstring>#include......
  • 对于数组和指针的关系的测试
    #include"stdio.h"//验证数组和指针的以下一些关系//1.一元数组名本质上是数组第一个元素的地址,也是数组的地址//2。数组中存在a[2]=*(a+2)//3.数组在传递的时候传递的是数组名,也就是传递的是它的地址intmain(){intc[3]={1,2,3};int*a=c;//此时的a表示c数组......
  • 洛谷4113(树状数组+离线处理)
    [HEOI2012]采花题目描述萧薰儿是古国的公主,平时的一大爱好是采花。今天天气晴朗,阳光明媚,公主清晨便去了皇宫中新建的花园采花。花园足够大,容纳了$n$朵花,共有$c$种颜色,用整数$1\simc$表示。且花是排成一排的,以便于公主采花。公主每次采花后会统计采到的花的颜色数,颜色......
  • 222. 完全二叉树的节点个数
    给你一棵完全二叉树的根节点root,求出该树的节点个数。完全二叉树的定义如下:在完全二叉树中,除了最底层节点可能没填满外,其余每层节点数都达到最大值,并且最下面一层的节点都集中在该层最左边的若干位置。若最底层为第h层,则该层包含1~2h个节点。classSolution{public:......
  • vue数组和对象进行 watch 和 watchEffect 对比
    constarr1=ref([]);constarr2=reactive([]);constobj1=ref({});constobj2=reactive({});watchEffect(()=>{console.log("watchEffectarr1",arr1.value);console.log("watchEffectarr2",arr2)......
  • PHP 判断数组是否为空的方法
    1.isset功能:判断变量是否被初始化说明:它并不会判断变量是否为空,并且可以用来判断数组中元素是否被定义过注意:当使用isset来判断数组元素是否被初始化过时,它的效率比array_key_exists高4倍左右<?php$a='';$a['c']='';if(!isset($a))echo'$a未被初始化'."";if......
  • JavaScript:数组删除指定元素
    1.shift()方法用于删除数组中的第一个元素。注:此方法会改变数组的长度letarr=[1,2,3]arr.shift()//删除1//arr为[2,3]2.pop()方法用于删除数组中最后一个元素注:此方法会改变数组的长度letarr=[1,2,3]arr.pop();//删除3//arr为[1,2]3.splice()方法用于......
  • 力扣-数组-滑动窗口
    题目顺序209长度最小的子数组,904水果成篮解题思路1.滑动窗口求解的题目中,关键词为”求解连续“2.暴力解法是双重for循环,相当于对滑动窗口的起始和终止点都遍历3.滑动窗口求解是,只遍历终止点,当sum符合条件时,start++,向前一步缩小窗口4.终止条件是终止点end遍历完  1c......
  • 【LeetCode排序专题02】最小k个数,关于快速排序的讨论
    最小k个数输入整数数组arr,找出其中最小的k个数。例如,输入4、5、1、6、2、7、3、8这8个数字,则最小的4个数字是1、2、3、4。示例1:输入:arr=[3,2,1],k=2输出:[1,2]或者[2,1]示例2:输入:arr=[0,1,2,1],k=1输出:[0]限制:0<=k<=arr.length<=100000<=arr[i......
  • js 递归遍历树形结构数据,返回新的数组
    工作中,我们经常会遇到这样的情况:后端返回的数组,只需要取name、value生成新的数组,或者是将某个属性名修改,生成新的数组。递归是一种常见的解决问题的方法,即把问题逐渐简单化。“递归”的基本思想是:自己调用自己。实例如下handleDg(arrs,that){arrs.map((item,index)......