Leetcode.2055 蜡烛之间的盘子

题目链接

Leetcode.2055 蜡烛之间的盘子 rating : 1819

题目描述

给你一个长桌子,桌子上盘子和蜡烛排成一列。给你一个下标从 0 0 0 开始的字符串 s s s ,它只包含字符 '*''|' ,其中 '*' 表示一个 盘子 ,'|' 表示一支 蜡烛 。

同时给你一个下标从 0 0 0 开始的二维整数数组 q u e r i e s queries queries ,其中 q u e r i e s [ i ] = [ l e f t i , r i g h t i ] queries[i] = [lefti, righti] queries[i]=[lefti,righti] 表示 子字符串 s [ l e f t i . . . r i g h t i ] s[lefti...righti] s[lefti...righti] (包含左右端点的字符)。对于每个查询,你需要找到 子字符串中两支蜡烛之间 的盘子的 数目 。如果一个盘子在 子字符串中 左边和右边至少有一支蜡烛,那么这个盘子满足在 两支蜡烛之间

  • 比方说,s = "||**||**|*" ,查询 [ 3 , 8 ] [3, 8] [3,8] ,表示的是子字符串 "*||**|" 。子字符串中在两支蜡烛之间的盘子数目为 2 2 2 ,子字符串中右边两个盘子在它们左边和右边 至少有一支蜡烛。

请你返回一个整数数组 a n s w e r answer answer,其中 a n s w e r [ i ] answer[i] answer[i] 是第 i i i 个查询的答案。

示例 1:

在这里插入图片描述

输入:s = “||***|”, queries = [[2,5],[5,9]]
输出:[2,3]
解释:

  • queries[0] 有两个盘子在蜡烛之间。
  • queries[1] 有三个盘子在蜡烛之间。
示例 2:

在这里插入图片描述

输入:s = “||||||*”, queries = [[1,17],[4,5],[14,17],[5,11],[15,16]]
输出:[9,0,0,0,0]
解释:

  • queries[0] 有 9 个盘子在蜡烛之间。
  • 另一个查询没有盘子在蜡烛之间。

提示:

  • 3 ≤ s . l e n g t h ≤ 1 0 5 3 \leq s.length \leq 10^5 3s.length105
  • s s s 只包含字符 '*''|'
  • 1 ≤ q u e r i e s . l e n g t h ≤ 1 0 5 1 \leq queries.length \leq 10^5 1queries.length105
  • q u e r i e s [ i ] . l e n g t h = = 2 queries[i].length == 2 queries[i].length==2
  • 0 ≤ l e f t i ≤ r i g h t i < s . l e n g t h 0 \leq lefti \leq righti < s.length 0leftirighti<s.length

解法:前缀和

p r e s u m presum presum 记录蜡烛的前缀和数量。例如 p r e s u m [ r ] − p r e s u m [ l − 1 ] presum[r] - presum[l-1] presum[r]presum[l1] 就是区间 [ l , r ] [l,r] [l,r] 内的盘子数量。

l e f t left left 记录每一个位置左边蜡烛的位置。例如 l e f t [ i ] left[i] left[i] 就是位置 i i i 左边最近的蜡烛的位置,没有的话默认为 − 1 -1 1

r i g h t right right 记录每一个位置右边蜡烛的位置。例如 r i g h t [ i ] right[i] right[i] 就是位置 i i i 右边最近的蜡烛的位置,没有的话默认为 n n n

对于每一个查询 q u e r i e s [ i ] = [ a , b ] queries[i] = [a,b] queries[i]=[a,b],设 l = r i g h t [ a ] , r = l e f t [ b ] l = right[a] , r = left[b] l=right[a],r=left[b] [ l , r ] [l,r] [l,r] 就是我们要计算盘子的有效范围,如果 l ≥ r l \geq r lr,说明区间 [ a , b ] [a,b] [a,b] 内没有有效的盘子;否则,区间 [ a , b ] [a,b] [a,b] 之内的有效盘子数量为 : p r e s u m [ r ] − p r e s u m [ l ] presum[r] - presum[l] presum[r]presum[l]

时间复杂度: O ( n ) O(n) O(n)

C++代码:

class Solution {
public:
    vector<int> platesBetweenCandles(string s, vector<vector<int>>& queries) {
        int n = s.size();
        vector<int> presum(n);
        for(int i = 0,sum = 0;i < n;i++){
            if(s[i] == '*') sum++;
            presum[i] = sum;
        }

        vector<int> left(n , -1);
        for(int i = 0,pre = -1;i < n;i++){
            if(s[i] == '|') pre = i;
            left[i] = pre;
        }

        vector<int> right(n,n);
        for(int i = n - 1,pre = n;i >= 0;--i){
            if(s[i] == '|') pre = i;
            right[i] = pre;
        }

        int m = queries.size();
        vector<int> ans(m);

        for(int i = 0;i < m;i++){
            int a = queries[i][0] , b = queries[i][1];
            int l = right[a] , r = left[b];
            if(l >= r) continue;
            ans[i] = presum[r] - presum[l];
        }
        

        return ans;
    }
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值