给你一个字符串 s
。请返回 s
中最长的 超赞子字符串 的长度。
「超赞子字符串」需满足满足下述两个条件:
- 该字符串是
s
的一个非空子字符串 - 进行任意次数的字符交换后,该字符串可以变成一个回文字符串
示例 1:
输入:s = "3242415" 输出:5 解释:"24241" 是最长的超赞子字符串,交换其中的字符后,可以得到回文 "24142"
示例 2:
输入:s = "12345678" 输出:1
示例 3:
输入:s = "213123" 输出:6 解释:"213123" 是最长的超赞子字符串,交换其中的字符后,可以得到回文 "231132"
示例 4:
输入:s = "00" 输出:2
提示:
1 <= s.length <= 10^5
s
仅由数字组成
方法一:状态压缩 + 前缀和思想
根据题目描述,“超赞子字符串”中的字符可以通过交换得到回文字符串,因此,“超赞子字符串”中最多有一个数字字符出现奇数次,其余数字字符出现偶数次。
我们可以用一个整数
而如果子字符串
所以,我们可以用哈希表或数组记录所有状态
最后,返回答案即可。
时间复杂度
class Solution:
def longestAwesome(self, s: str) -> int:
st = 0
d = {0: -1}
ans = 1
for i, c in enumerate(s):
v = int(c)
st ^= 1 << v
if st in d:
ans = max(ans, i - d[st])
else:
d[st] = i
for v in range(10):
if st ^ (1 << v) in d:
ans = max(ans, i - d[st ^ (1 << v)])
return ans
class Solution {
public int longestAwesome(String s) {
int[] d = new int[1024];
int st = 0, ans = 1;
Arrays.fill(d, -1);
d[0] = 0;
for (int i = 1; i <= s.length(); ++i) {
int v = s.charAt(i - 1) - '0';
st ^= 1 << v;
if (d[st] >= 0) {
ans = Math.max(ans, i - d[st]);
} else {
d[st] = i;
}
for (v = 0; v < 10; ++v) {
if (d[st ^ (1 << v)] >= 0) {
ans = Math.max(ans, i - d[st ^ (1 << v)]);
}
}
}
return ans;
}
}
class Solution {
public:
int longestAwesome(string s) {
vector<int> d(1024, -1);
d[0] = 0;
int st = 0, ans = 1;
for (int i = 1; i <= s.size(); ++i) {
int v = s[i - 1] - '0';
st ^= 1 << v;
if (~d[st]) {
ans = max(ans, i - d[st]);
} else {
d[st] = i;
}
for (v = 0; v < 10; ++v) {
if (~d[st ^ (1 << v)]) {
ans = max(ans, i - d[st ^ (1 << v)]);
}
}
}
return ans;
}
};
func longestAwesome(s string) int {
d := [1024]int{}
d[0] = 1
st, ans := 0, 1
for i, c := range s {
i += 2
st ^= 1 << (c - '0')
if d[st] > 0 {
ans = max(ans, i-d[st])
} else {
d[st] = i
}
for v := 0; v < 10; v++ {
if d[st^(1<<v)] > 0 {
ans = max(ans, i-d[st^(1<<v)])
}
}
}
return ans
}
func max(a, b int) int {
if a > b {
return a
}
return b
}