【LeetCode贪心#10】划分字母区间(有涉及hash数组的使用)
划分字母区间
字符串 S 由小写字母组成。我们要把这个字符串划分为尽可能多的片段,同一字母最多出现在一个片段中。返回一个表示每个字符串片段的长度的列表。
示例:
- 输入:S = "ababcbacadefegdehijhklij"
- 输出:[9,7,8] 解释: 划分结果为 "ababcbaca", "defegde", "hijhklij"。 每个字母最多出现在一个片段中。 像 "ababcbacadefegde", "hijhklij" 的划分是错误的,因为划分的片段数较少。
提示:
- S的长度在[1, 500]之间。
- S只包含小写字母 'a' 到 'z' 。
思路
题目分析
题目是什么意思呢?拿示例来说
当前输入为:S = "ababcbacadefegdehijhklij"
ababcbacadefegdehijhklij
↑
此时遍历字符串,取到子串'a',那么S中的所有'a'都要包含到当前子串中,于是继续往后遍历
ababcbacadefegdehijhklij
↑
此时又取到'b',同理S中的所有'b'都要包含到当前子串中,当前子串为"ab"
以此类推,当遍历一段时间后,可以划分出第一个片段,即"ababcbaca"
ababcbacadefegdehijhklij
↑
记录字符出现位置
上述过程中一个核心问题是:如何确定当前字母在字符串中的最远位置
只有解决这个问题才能获取片段的结束位置
这里通过哈希表的方式来解决(此技巧在 有效的字母异位词 也出现过)
定义一个hash数组 int hash[27] = {0}
,有26个字母,所以数组大小设为27就很够了
然后遍历字符串S
int hash[27] = {0};
for(int i = 0; i < S.size(); ++i){
}
同时,记录当前遍历所得的字符的位置
int hash[27] = {0};
for(int i = 0; i < S.size(); ++i){
hash[S[i] - 'a'] = i;//
}
S[i]
是当前字符,通过S[i] - 'a'
确定其是第几个字母,并得到其在hash数组中的具体位置
然后再hash数组中记录当前遍历过程中S[i]
在S中出现的位置 i
例如:第一个遍历得到的字符是'b',此时可以定位到其位于hash数组中的 下标1 位置(看做字母表的话就是第二个)
i 代表着'b'目前在S中的第 i 个位置出现,记录下来即可
通过遍历过程的不断推进,hash数组会记录下某个字符最后一次在S中出现的位置,这便是最远位置
由上述方法我们就可以记录任意26个字母在某个字符串中出现过的最远位置
切片
在代码实现时,我们会先遍历一遍S,将每个字符在S中的最远位置求出来记录于hash数组
此时,我们再次遍历S,设置左右边界,在遍历过程中不断更新右边界
直到当前遍历下标i与右边界相等,此时意味着我们已经找到分割位置,然后可以将结果保存(结果为分割子串的长度,因此只需保存左右边界的作差结果即可)
借用一下卡哥的图
如图,第一次遍历完S之后得到hash数组,由该数组显示S的第一个字符'a'最后一次出现位置是 下标8 处
当第二次遍历S时,使用遍历下标 i 为hash数组的索引,不断地取当前遍历字符的最远出现位置来更新右边界right
而更新时只用最大值来更新,因为'a'的最远位置是8,可以看到"ababcbaca"中没有哪个字符的位置比8要远,因此右边界就一直指向8
那么,当i == right
时,我们就计算并保存"ababcbaca"的长度,即right - left + 1
for (int i = 0; i < S.size(); i++) {
right = max(right, hash[S[i] - 'a']); // 找到字符出现的最远边界
if (i == right) {
result.push_back(right - left + 1);
left = i + 1;
}
}
代码
根据上面的分析,得到步骤如下:
1、创建一个hash数组(需要记录某个字符出现的最远位置作为分割依据)
2、遍历目标字符串,记录每个字符出现的最远位置
3、定义左右边界遍历与结果变量
4、再次遍历字符串,使用下标i去hash数组中找当前遍历字符的最远距离,不断更新右边界right
5、当i == right,记录结果
class Solution {
public:
vector<int> partitionLabels(string s) {
//定义hash数组,记录每个字符的最远位置
int hash[27] = {0};
for(int i = 0; i < s.size(); ++i){//获取每个字符的最远位置
hash[s[i] - 'a'] = i;//记录当前字母在s中出现的位置i
}
vector<int> res;
int left = 0;//定义左右边界
int right = 0;
for(int i = 0; i < s.size(); ++i){
//取hash中的最大值更新right
right = max(hash[s[i] - 'a'], right);
if(i == right){
res.push_back(right - left + 1);
left = i + 1;//记录一个片段后更新左边界
}
}
return res;
}
};
注意点:
在 C++ 中,int a, b = 0
表示定义了两个整型变量 a
和 b
,其中 b
被初始化为 0,而 a
没有被初始化,其值是未定义的。这种语法是 C++ 中的变量声明的一种简写方式,但容易造成误解,应尽量避免使用。
而 int a = 0; int b = 0;
则表示定义了两个整型变量 a
和 b
,它们都被初始化为 0。这种写法更加明确,不会造成误解,建议优先使用。
因此,这两种写法的区别在于变量初始化的方式和是否对变量进行了初始化。在编写代码时,应该尽量避免使用第一种写法,而是采用第二种写法进行变量定义和初始化。
【LeetCode贪心#10】划分字母区间(有涉及hash数组的使用)的更多相关文章
- LeetCode:划分字母区间【763】
LeetCode:划分字母区间[763] 题目描述 字符串 S 由小写字母组成.我们要把这个字符串划分为尽可能多的片段,同一个字母只会出现在其中的一个片段.返回一个表示每个字符串片段的长度的列表. 示 ...
- Java实现 LeetCode 763 划分字母区间(暴力)
763. 划分字母区间 字符串 S 由小写字母组成.我们要把这个字符串划分为尽可能多的片段,同一个字母只会出现在其中的一个片段.返回一个表示每个字符串片段的长度的列表. 示例 1: 输入: S = & ...
- 【LeetCode】763-划分字母区间
title: 763-划分字母区间 date: 2019-04-15 21:10:46 categories: LeetCode tags: 字符串 贪心思想 双指针 题目描述 字符串 S 由小写字母 ...
- LeetCode 763划分字母区间 详解
题目详情 字符串 S 由小写字母组成.我们要把这个字符串划分为尽可能多的片段,同一个字母只会出现在其中的一个片段.返回一个表示每个字符串片段的长度的列表. 示例 1: 输入: S = "ab ...
- leetcode 763. 划分字母区间
题目描述: 字符串 S 由小写字母组成.我们要把这个字符串划分为尽可能多的片段,同一个字母只会出现在其中的一个片段.返回一个表示每个字符串片段的长度的列表. 示例 1: 输入: S = "a ...
- JS 之 每日一题 之 算法 ( 划分字母区间 )
题目详解: 字符串 S 由小写字母组成.我们要把这个字符串划分为尽可能多的片段,同一个字母只会出现在其中的一个片段.返回一个表示每个字符串片段的长度的列表. 例子: 示例 1: 输入:S = &quo ...
- [Swift]LeetCode763. 划分字母区间 | Partition Labels
A string S of lowercase letters is given. We want to partition this string into as many parts as pos ...
- [LeetCode] Shifting Letters 漂移字母
We have a string S of lowercase letters, and an integer array shifts. Call the shift of a letter, th ...
- LeetCode 57. Insert Interval 插入区间 (C++/Java)
题目: Given a set of non-overlapping intervals, insert a new interval into the intervals (merge if nec ...
- hdu6003 Problem Buyer 贪心 给定n个区间,以及m个数,求从n个区间中任意选k个区间,满足m个数都能在k个区间中找到一个包含它的区间,如果一个区间包含了x,那么 该区间不能再去包含另一个数,即k>=m。求最小的k。如果不存在这样的k,输出“IMPOSSIBLE!”。
/** 题目:hdu6003 Problem Buyer 链接:http://acm.hdu.edu.cn/showproblem.php?pid=6003 题意:给定n个区间,以及m个数,求从n个区 ...
随机推荐
- Oracle 核心列信息查看与处理
Oracle 核心列信息查看与处理 背景 最近想对数据库表进行跨数据之间的比照 因为有一些自增列或者是时间戳的列不需要进行对比 后者是对比容易导致失真. 所以就准备选用其他方式进行一下处理. 本文主要 ...
- Booking.com如何在毫秒内搜索数百万个地点
译自:How Booking.com Searches Through Millions of Locations in Milliseconds Booking.com是一家与酒店.旅馆.度假租赁等 ...
- 01uni-app的创建运行在不同端上的配置 以及tarBar的配置
uni-app的创建### 01==>创建uni-app的项目非常简单.不需要注意什么注意点哈!! 创建项目的时候 可以参考官网 https://uniapp.dcloud.io/quickst ...
- vue如何获取动态添加的类
动态添加的类.你在声明周期中的mounted中是拿不到的. 是有在updata这个声明周期中才可以拿到的. 因为此时数据才跟新完成
- 【K哥爬虫普法】微信公众号爬虫构成不正当竞争,爬虫er面对金山,如何避免滥用爬虫?
我国目前并未出台专门针对网络爬虫技术的法律规范,但在司法实践中,相关判决已屡见不鲜,K 哥特设了"K哥爬虫普法"专栏,本栏目通过对真实案例的分析,旨在提高广大爬虫工程师的法律意识, ...
- jenkins 安装与构建过程中的系列问题
一.插件安装遇到的依赖问题 插件安装分为在线安装和离线安装 1.在线安装 搜索要安装的插件,然后进行安装即可 2.离线安装hpi文件 使用该方法安装插件每次只能安装一个插件,且如果插件之间存在依赖性则 ...
- kettle系统列文章02---如何建立一个转换
1.连接mysql 主对象树---->DB连接---->新建 2.连接sqlserver 主对象树--->DB连接----->新建 3.设置数据库为共享:在db上右键---&g ...
- 7.3 Windows驱动开发:内核监视LoadImage映像回调
在笔者上一篇文章<内核注册并监控对象回调>介绍了如何运用ObRegisterCallbacks注册进程与线程回调,并通过该回调实现了拦截指定进行运行的效果,本章LyShark将带大家继续探 ...
- C/C++ 通过HTTP实现文件上传下载
WinInet(Windows Internet)是 Microsoft Windows 操作系统中的一个 API 集,用于提供对 Internet 相关功能的支持.它包括了一系列的函数,使得 Win ...
- C++ Boost库 实现命令行解析
Boost库中默认自带了一个功能强大的命令行参数解析器,以往我都是自己实现参数解析的,今天偶尔发现这个好东西,就来总结一下参数解析的基本用法,该库需要引入program_options.hpp头文件, ...