给你一个由小写字母组成的字符串 s
,和一个整数 k
。如果满足下述条件,则可以将字符串 t
视作是 理想字符串 :
t
是字符串s
的一个子序列。t
中每两个 相邻 字母在字母表中位次的绝对差值小于或等于k
。
返回 最长 理想字符串的长度。
字符串的子序列同样是一个字符串,并且子序列还满足:可以经由其他字符串删除某些字符(也可以不删除)但不改变剩余字符的顺序得到。
注意:字母表顺序不会循环。例如,'a'
和 'z'
在字母表中位次的绝对差值是 25
,而不是 1
。
示例 1:
输入:s = "acfgbd", k = 2 输出:4 解释:最长理想字符串是 "acbd" 。该字符串长度为 4 ,所以返回 4 。 注意 "acfgbd" 不是理想字符串,因为 'c' 和 'f' 的字母表位次差值为 3 。
示例 2:
输入:s = "abcd", k = 3 输出:4 解释:最长理想字符串是 "abcd" ,该字符串长度为 4 ,所以返回 4 。
提示:
1 <= s.length <= 105
0 <= k <= 25
s
由小写英文字母组成
方法一:动态规划
设
在
答案为
时间复杂度
class Solution:
def longestIdealString(self, s: str, k: int) -> int:
n = len(s)
ans = 1
dp = [1] * n
d = {s[0]: 0}
for i in range(1, n):
a = ord(s[i])
for b in ascii_lowercase:
if abs(a - ord(b)) > k:
continue
if b in d:
dp[i] = max(dp[i], dp[d[b]] + 1)
d[s[i]] = i
return max(dp)
class Solution {
public int longestIdealString(String s, int k) {
int n = s.length();
int ans = 1;
int[] dp = new int[n];
Arrays.fill(dp, 1);
Map<Character, Integer> d = new HashMap<>(26);
d.put(s.charAt(0), 0);
for (int i = 1; i < n; ++i) {
char a = s.charAt(i);
for (char b = 'a'; b <= 'z'; ++b) {
if (Math.abs(a - b) > k) {
continue;
}
if (d.containsKey(b)) {
dp[i] = Math.max(dp[i], dp[d.get(b)] + 1);
}
}
d.put(a, i);
ans = Math.max(ans, dp[i]);
}
return ans;
}
}
class Solution {
public:
int longestIdealString(string s, int k) {
int n = s.size();
int ans = 1;
vector<int> dp(n, 1);
unordered_map<char, int> d;
d[s[0]] = 0;
for (int i = 1; i < n; ++i) {
char a = s[i];
for (char b = 'a'; b <= 'z'; ++b) {
if (abs(a - b) > k) continue;
if (d.count(b)) dp[i] = max(dp[i], dp[d[b]] + 1);
}
d[a] = i;
ans = max(ans, dp[i]);
}
return ans;
}
};
func longestIdealString(s string, k int) int {
n := len(s)
ans := 1
dp := make([]int, n)
for i := range dp {
dp[i] = 1
}
d := map[byte]int{s[0]: 0}
for i := 1; i < n; i++ {
a := s[i]
for b := byte('a'); b <= byte('z'); b++ {
if int(a)-int(b) > k || int(b)-int(a) > k {
continue
}
if v, ok := d[b]; ok {
dp[i] = max(dp[i], dp[v]+1)
}
}
d[a] = i
ans = max(ans, dp[i])
}
return ans
}
func max(a, b int) int {
if a > b {
return a
}
return b
}
function longestIdealString(s: string, k: number): number {
const dp = new Array(26).fill(0);
for (const c of s) {
const x = c.charCodeAt(0) - 'a'.charCodeAt(0);
let t = 0;
for (let i = 0; i < 26; i++) {
if (Math.abs(x - i) <= k) {
t = Math.max(t, dp[i] + 1);
}
}
dp[x] = Math.max(dp[x], t);
}
return dp.reduce((r, c) => Math.max(r, c), 0);
}