来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/plates-between-candles

题目描述

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

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

比方说,s = "||**||**|*" ,查询 [3, 8] ,表示的是子字符串 "*||**|" 。子字符串中在两支蜡烛之间的盘子数目为 2 ,子字符串中右边两个盘子在它们左边和右边 都 至少有一支蜡烛。
请你返回一个整数数组 answer ,其中 answer[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.length <= 105
s 只包含字符 '*' 和 '|' 。
1 <= queries.length <= 105
queries[i].length == 2
0 <= lefti <= righti < s.length

解题思路

最初的想法是将蜡烛位置全部记录下来,然后将区间[a,b]转换成其最大蜡烛子区间[c,d],其中盘子数就是d - c - id-ic 其中,id、ic是蜡烛d c在蜡烛中的排序,时间复杂度为O(n + n * m)时间超时了

之后使用前缀和的思想,将盘子的前缀和左边右边第一个蜡烛分别记录下来,区间[a,b]的盘子数就是区间[c,d]的盘子数,也就是sumd - sumc,时间复杂度为O(n + m)

第一种解法可以进一步优化,使用二分法查找c和d时间复杂度会降为O(n + m * log(n)).

这里使用了两个库函数lower_bound 和 upper_bound,这两个库函数是属于<algorithm>头文件中,底层实现基于二分查找,lower_bound是查找数组中大于等于x的第一个数,而upper_bound是查找大于x的第一个数。

新增不使用库函数的二分写法。

代码展示

前缀和

class Solution {
public:
vector<int> platesBetweenCandles(string s, vector<vector<int>>& queries) {
vector<int> viRet;
int n = s.size();
vector<int> viSum(n);
int sum = 0;
for(int i =0; i < n; i++)
{
if(s[i] == '*')
{
sum++;
}
viSum[i] = sum;
}
vector<int> viLeft(n);
for(int i =0, l = -1; i < n; i++)
{
if(s[i] == '|')
{
l = i;
}
viLeft[i] = l;
}
vector<int> viRight(n);
for(int i = n - 1, l = -1; i >= 0; i--)
{
if(s[i] == '|')
{
l = i;
}
viRight[i] = l;
} for(auto iter: queries)
{
int iLeft = viRight[iter[0]], iRight = viLeft[iter[1]];
if(iLeft == -1 || iRight == -1 || iLeft >= iRight)
{
viRet.push_back(0);
}
else
{
viRet.push_back(viSum[iRight] - viSum[iLeft]);
}
}
return viRet;
}
};

预处理+库函数二分:

class Solution {
public:
vector<int> platesBetweenCandles(string s, vector<vector<int>>& queries) {
vector<int> viRet;
vector<int> viIndex;
for(int i =0; i < s.size(); i ++)
{
if(s[i] == '|')
{
viIndex.push_back(i);
}
}
for(auto iter: queries)
{
int iLeft = -1, iLeftIndex = -1, iRight = -1, iRightIndex = -1;
iLeftIndex = lower_bound(viIndex.begin(), viIndex.end(), iter[0]) - viIndex.begin();
iRightIndex = upper_bound(viIndex.begin(), viIndex.end(), iter[1]) - viIndex.begin();
iRightIndex--;
if(iLeftIndex == -1 || iRightIndex == -1 || iLeftIndex >= iRightIndex)
{
viRet.push_back(0);
}
else
{
iLeft = viIndex[iLeftIndex];
iRight = viIndex[iRightIndex];
viRet.push_back(iRight - iLeft - iRightIndex + iLeftIndex);
}
}
return viRet;
}
};

预处理+二分

class Solution {
public:
vector<int> platesBetweenCandles(string s, vector<vector<int>>& queries) {
vector<int> viRet(queries.size(), 0);
vector<int> viIndex;
for (int i = 0; i < s.size(); i++)
{
if (s[i] == '|')
{
viIndex.push_back(i);
}
}
if(!viIndex.size()) return viRet;
for (int i = 0; i < queries.size(); i++)
{
int iLeft = -1, iLeftIndex = -1, iRight = -1, iRightIndex = -1;
int l = 0, r = viIndex.size() - 1;
while (l < r)
{
int mid = (l + r) / 2;
if (viIndex[mid] >= queries[i][0])
{
r = mid;
}
else
{
l = mid + 1;
}
}
if (viIndex[r] >= queries[i][0])
{
iLeft = viIndex[r];
iLeftIndex = r;
}
else
continue;
l = 0, r = viIndex.size() - 1;
while (l < r)
{
int mid = (l + r + 1) / 2;
if (viIndex[mid] <= queries[i][1])
{
l = mid;
}
else
{
r = mid - 1;
}
}
if (viIndex[r] <= queries[i][1])
{
iRight = viIndex[r];
iRightIndex = r;
}
else
continue;
if (iLeft == -1 || iRight == -1 || iLeft >= iRight)
{ }
else
{
viRet[i] = iRight - iLeft - iRightIndex + iLeftIndex;
}
}
return viRet;
}
};

运行结果

LeetCode-2055 蜡烛之间的盘子 及库函数 lower_bound 和 upper_bound学习使用的更多相关文章

  1. [动态规划] LeetCode 2055. 蜡烛之间的盘子

    LeetCode 2055 蜡烛之间的盘子 前言: 这个题做的时间略长了,开始的时候打算先定位两个端点的蜡烛,之后在遍历其中的盘子,结果不言而喻,必time limit了,之后就预处理了前x的蜡烛间盘 ...

  2. LeetCode 34 - 在排序数组中查找元素的第一个和最后一个位置 - [二分][lower_bound和upper_bound]

    给定一个按照升序排列的整数数组 nums,和一个目标值 target.找出给定目标值在数组中的开始位置和结束位置. 你的算法时间复杂度必须是 O(log n) 级别. 如果数组中不存在目标值,返回 [ ...

  3. [LeetCode] Count of Range Sum 区间和计数

    Given an integer array nums, return the number of range sums that lie in [lower, upper] inclusive.Ra ...

  4. leetcode bugfree note

    463. Island Perimeterhttps://leetcode.com/problems/island-perimeter/就是逐一遍历所有的cell,用分离的cell总的的边数减去重叠的 ...

  5. LeetCode OJ 题解

    博客搬至blog.csgrandeur.com,cnblogs不再更新. 新的题解会更新在新博客:http://blog.csgrandeur.com/2014/01/15/LeetCode-OJ-S ...

  6. leetcode:Roman to Integer(罗马数字转化为罗马数字)

    Question: Given a roman numeral, convert it to an integer. Input is guaranteed to be within the rang ...

  7. [LeetCode] Falling Squares 下落的方块

    On an infinite number line (x-axis), we drop given squares in the order they are given. The i-th squ ...

  8. C++引用C程序库函数

    C与C++混合编程 C++里面如何声明const void f(void)函数为C程序中的库函数. void f(void)用c++ compiler来编译,在产生的obj文件中的名字变成了 $f@@ ...

  9. [LeetCode] 327. Count of Range Sum 区间和计数

    Given an integer array nums, return the number of range sums that lie in [lower, upper] inclusive.Ra ...

  10. 【LeetCode】数学(共106题)

    [2]Add Two Numbers (2018年12月23日,review) 链表的高精度加法. 题解:链表专题:https://www.cnblogs.com/zhangwanying/p/979 ...

随机推荐

  1. 搭建IIS网站后,点击浏览地址,报403错误

    点击左侧的浏览地址,报右侧的错误,可将目录浏览进行启用 双击进去,进行启用即可

  2. Jmeter 之bzm- Concurrency Thread Group 压测

    bzm- Concurrency Thread Group  并发线程组代替 jp@gc - Stepping Thread Group线程组. 1.  下载jmeter-plugins-manage ...

  3. pyCharm中下载包的速度慢的解决方案

    1.解决方案 使用阿里镜像 2.具体步骤 1.在项目里面新建一个xxx.py文件 2.然后将下面的代码复制进xxx.py文件 import os ini = "[global]\nindex ...

  4. 网络监测工具之Zabbix的搭建与测试方法(二)-- SNMP、OID和MIB概述

    概念 SNMP是专门设计用于在 IP 网络管理网络节点的一种标准协议,它是一种应用层协议.SNMP使网络管理员能够管理网络效能,发现并解决网络问题以及规划网络增长.通过SNMP接收随机消息(及事件报告 ...

  5. 【集成开发环境 (IDE)】Dev-Cpp下载与安装 [ 图文教程 ]

    版权声明 本文作者:main工作室 本文链接:https://www.cnblogs.com/main-studio/p/17037280.html 版权声明:本文为 博客园 博主「main工作室」的 ...

  6. MAC上好用的解压工具

    macOS:11.1 想在 macOS 上打开一个压缩文件,有原生的归档实用工具或 BetterZip.Keka.The Unarchiver 等诸多选择.最近,又有国内独立开发者为我们带来了一款新作 ...

  7. Quartz 使用教程

    首先说说,为什么要写这篇文章: Quartz 的 v2.3.2 版本改动比较大,目前网上的资料都是旧版本,很缺乏相关资料 很多资料讲解非常不全面,例如 Quartz Listener 的介绍和使用基本 ...

  8. 第一个C程序

    写C代码 创建工作区 创建工程 创建文件(.c源文件..h头文件) main函数是程序执行的入口,有且只有一个 函数printf(print function打印函数)[库函数-C语言本身提供给我们使 ...

  9. SpringBoot项目动态定时任务之 ScheduledTaskRegistrar(解决方案一)

    前言 ​ 在做SpringBoot项目的过程中,有时客户会提出按照指定时间执行一次业务的需求. ​ 如果客户需要改动业务的执行时间,即动态地调整定时任务的执行时间,那么可以采用SpringBoot自带 ...

  10. MySQL 合并查询union 查询出的行合并到一个表中

    在合并查询中,尤其是二分类的情况,在查询结果是相同列名的时候可以考虑合并查询.先查询出行的结果,再使用union或者union all合并查询结果. 另外如果 union 和 order by 一起使 ...