划分字母区间
Leetcode 763
学习记录自代码随想录
给你一个字符串 s 。我们要把这个字符串划分为尽可能多的片段,同一字母最多出现在一个片段中。
注意,划分结果需要满足:将所有划分结果按顺序连接,得到的字符串仍然是 s 。
返回一个表示每个字符串片段的长度的列表。
示例 1:
输入:s = “ababcbacadefegdehijhklij”
输出:[9,7,8]
解释:
划分结果为 “ababcbaca”、“defegde”、“hijhklij” 。
每个字母最多出现在一个片段中。
像 “ababcbacadefegde”, “hijhklij” 这样的划分是错误的,因为划分的片段数较少。
示例 2:
输入:s = “eccbbbbdec”
输出:[10]
提示:
1 <= s.length <= 500
s 仅由小写英文字母组成
要点:1.遍历过程中寻找每个字母的边界值,若找到之前遍历字母的最大边界值则意味着找到分割点;
2.实际上将每个字母的左右区间找到后,就可将问题转化为区间问题;
方法一:
统计每一个字符最后出现的位置
从头遍历字符,并更新字符的最远出现下标,如果找到字符最远出现位置下标和当前下标相等了,则找到了分割点
class Solution {
public:vector<int> partitionLabels(string s) {int hash[26] = {0};vector<int> result;for(int i = 0; i < s.size(); i++){hash[s[i] - 'a'] = i;}int left = 0;int right = 0;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;}}return result;}
};
方法二:将每个字母的左右边界区间找到并存入一个二维数组中,之后对数组进行排序,之后对其遍历寻找区间分割点,即与无重叠区间思路相似
class Solution{
private:vector<vector<int>> countLabels(string s){vector<vector<int>> hash(26, vector<int>(2, INT_MIN));vector<vector<int>> result;for(int i = 0; i < s.size(); i++){if(hash[s[i]-'a'][0] == INT_MIN){hash[s[i]-'a'][0] = i;}hash[s[i]-'a'][1] = i;}for(int i = 0; i < hash.size(); i++){if(hash[i][0] != INT_MIN){result.push_back(hash[i]);}}return result;}public:vector<int> partitionLabels(string s){vector<int> result;vector<vector<int>> hash = countLabels(s);if(hash.size() == 1) return {hash[0][1]+1};sort(hash.begin(), hash.end(), [](auto& a, auto& b){return a[0] < b[0];});int left = 0;int right = hash[0][1];for(int i = 1; i < hash.size(); i++){// if(hash[i][0] < hash[i-1][1]){// hash[i][1] = max(hash[i][1], hash[i-1][1]);// }// if(hash[i][0] > hash[i-1][1]){if(hash[i][0] > right){result.push_back(right-left+1);left = hash[i][0];}right = max(right, hash[i][1]);}result.push_back(right-left+1);return result;}
};