LeetCode-2055 蜡烛之间的盘子 及库函数 lower_bound 和 upper_bound学习使用
来源:力扣(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学习使用的更多相关文章
- [动态规划] LeetCode 2055. 蜡烛之间的盘子
LeetCode 2055 蜡烛之间的盘子 前言: 这个题做的时间略长了,开始的时候打算先定位两个端点的蜡烛,之后在遍历其中的盘子,结果不言而喻,必time limit了,之后就预处理了前x的蜡烛间盘 ...
- LeetCode 34 - 在排序数组中查找元素的第一个和最后一个位置 - [二分][lower_bound和upper_bound]
给定一个按照升序排列的整数数组 nums,和一个目标值 target.找出给定目标值在数组中的开始位置和结束位置. 你的算法时间复杂度必须是 O(log n) 级别. 如果数组中不存在目标值,返回 [ ...
- [LeetCode] Count of Range Sum 区间和计数
Given an integer array nums, return the number of range sums that lie in [lower, upper] inclusive.Ra ...
- leetcode bugfree note
463. Island Perimeterhttps://leetcode.com/problems/island-perimeter/就是逐一遍历所有的cell,用分离的cell总的的边数减去重叠的 ...
- LeetCode OJ 题解
博客搬至blog.csgrandeur.com,cnblogs不再更新. 新的题解会更新在新博客:http://blog.csgrandeur.com/2014/01/15/LeetCode-OJ-S ...
- leetcode:Roman to Integer(罗马数字转化为罗马数字)
Question: Given a roman numeral, convert it to an integer. Input is guaranteed to be within the rang ...
- [LeetCode] Falling Squares 下落的方块
On an infinite number line (x-axis), we drop given squares in the order they are given. The i-th squ ...
- C++引用C程序库函数
C与C++混合编程 C++里面如何声明const void f(void)函数为C程序中的库函数. void f(void)用c++ compiler来编译,在产生的obj文件中的名字变成了 $f@@ ...
- [LeetCode] 327. Count of Range Sum 区间和计数
Given an integer array nums, return the number of range sums that lie in [lower, upper] inclusive.Ra ...
- 【LeetCode】数学(共106题)
[2]Add Two Numbers (2018年12月23日,review) 链表的高精度加法. 题解:链表专题:https://www.cnblogs.com/zhangwanying/p/979 ...
随机推荐
- salesforce零基础学习(一百二十三)Transaction Security 浅入浅出
本篇参考: https://help.salesforce.com/s/articleView?id=sf.enhanced_transaction_security_policy_types.htm ...
- 侦察工具——Httrack
前言 web渗透学习笔记,实验环境为Metasploitable靶机上的DVWA.此随笔介绍Web渗透侦察工具Httrack Httrack 简介 Httrack能够克隆拷贝目标网站上的所有可访问.可 ...
- java线程基础知识整理
目录 线程基本概念 1.java实现线程 2.线程的生命周期 3.线程常用的方法 3.1.sleep() 3.2.interrupt方法 3.3.stop方法 4.线程调度 4.1.常见的线程调度模型 ...
- [深度学习] CCPD车牌数据集介绍
CCPD是一个大型的.多样化的.经过仔细标注的中国城市车牌开源数据集.CCPD数据集主要分为CCPD2019数据集和CCPD2020(CCPD-Green)数据集.CCPD2019数据集车牌类型仅有普 ...
- 【rabbitmq】单独配置某一个消费者手动ack,其他消费者自动ack
前言:博主才疏学浅,此方案仅供参考,如有更优方案请大佬评论区告知,十分感谢✿✿ヽ(°▽°)ノ✿ 问题背景:同一个服务中存在多个不同业务的rabbitmq的消费者,其中一个推送业务的消费者需要加死信队列 ...
- CLISP学习(二)
它是一门函数式语言,你要用函数的思维来思考. 只不过与数学的表达不同的是,数学里的函数是在括号外 f(x) ,而lisp是在括号内,以列表的形式(f x), cos(x) --> (cos x ...
- Java内存区域有哪些构成?
目录 前言 Java 内存区域 程序计数器 虚拟机栈 本地方法栈 堆 方法区 字符串常量池 运行时常量池 直接内存 小结 作者:小牛呼噜噜 | https://xiaoniuhululu.com 计算 ...
- 目标检测+双目测距——基于yolov5
效果 测试例子1 输入: 左图 右图 输出: 测试例子2 输入: 左图 右图 输出: 测试例子3 输入: 左图 右图 输出: 核心代码 基于yolov5-6.1版本和双目测距+点云--使用Middle ...
- 判断一个对象是否是数组的n个方法,typeOf不能判断引用类型对象
- 多行文字自动换行居中--实测好用的很OK
实测好用的很~~~ <!DOCTYPE html> <html lang="en"> <head> <meta charset=" ...