来源:力扣(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. 简易博客页面小项目 html css

    项目预览 代码 html: <!DOCTYPE html> <html lang="en"> <head> <meta charset=& ...

  2. python虚拟环境和venv的使用

    目录 1.环境与虚拟环境 2.查看帮助 3.--system-site-package 命令 4.创建虚拟环境 5.激活/关闭虚拟环境 6.保存和复制虚拟环境 7.改变虚拟环境所指向的真实python ...

  3. last-child可能你也会踩的坑

    旧文章从语雀迁移过来,原日期为2021-07-14 问题 当时写在写一个列表,列表每一项需要下面加下划线,最后一项不加下划线.第一时间,想到使用 :``last-child 这个伪类来实现. 当时的代 ...

  4. [OpenCV实战]46 在OpenCV下应用图像强度变换实现图像对比度均衡

    本文主要介绍基于图像强度变换算法来实现图像对比度均衡.通过图像对比度均衡能够抑制图像中的无效信息,使图像转换为更符合计算机或人处理分析的形式,以提高图像的视觉价值和使用价值.本文主要通过OpenCV ...

  5. [深度学习] ncnn编译使用

    文章目录 工程 ncnn工程编译使用(cpu) ncnn工程编译使用(vulkan) 参考 工程 ncnn工程编译使用(cpu) 在linux下建立如CMakeLists文件即可编译生成ncnn工程 ...

  6. 基于Java的高并发多线程分片断点下载

    基于Java的高并发多线程分片断点下载 首先直接看测试情况: 单线程下载72MB文件 7线程并发分片下载72MB文件: 下载效率提高2-3倍,当然以上测试结果还和设备CPU核心数.网络带宽息息相关. ...

  7. Java中Elasticsearch 实现分页方式(三种方式)

    目录 ES 简介 ES 的特点: 一.from + size 浅分页 二.scroll 深分页 scroll删除 三.search_after 深分页 ES 简介 Elasticsearch 是一个基 ...

  8. flask_apscheduler

    网上很多文章包括官网给出来的,更多的是执行的函数和初始话app在同一文件 按照app 路由 模型类 视图 往下延申,项目已经成型,所以上述的方法不适用. (不用操心模型类与app互相导入的错误)下面分 ...

  9. Java读取文件后文件被占用

    Java读取文件响应后文件一直被占用问题 原因: 由于是封装的函数,请求和响应对象都是 形参地址 虽然在此函数里关闭了输出流,但是由于有返回值,调用未结束,输出流无法提前关闭 解决: 1:调用函数后, ...

  10. 普冉PY32系列(一) PY32F0系列32位Cortex M0+ MCU简介

    目录 普冉PY32系列(一) PY32F0系列32位Cortex M0+ MCU简介 普冉PY32系列(二) Ubuntu GCC Toolchain和VSCode开发环境 PY32F0系列上市其实相 ...