有什么不对的地方欢迎大佬评论区评论
5468. 第 k 个缺失的正整数(遍历)
5469. K 次操作转变字符串(遍历记录)
5470. 平衡括号字符串的最少插入次数(压栈)
5485. 找出最长的超赞子字符串(位运算,前缀和)
给你一个 严格升序排列 的正整数数组 arr 和一个整数 k 。
请你找到这个数组里第 k 个缺失的正整数。
示例 1:
输入:arr = [2,3,4,7,11], k = 5
输出:9
解释:缺失的正整数包括 [1,5,6,8,9,10,12,13,…] 。第 5 个缺失的正整数为 9 。
示例 2:
输入:arr = [1,2,3,4], k = 2
输出:6
解释:缺失的正整数包括 [5,6,7,…] 。第 2 个缺失的正整数为 6 。
提示:
1 <= arr.length <= 1000
1 <= arr[i] <= 1000
1 <= k <= 1000
对于所有 1 <= i < j <= arr.length 的 i 和 j 满足 arr[i] < arr[j]
class Solution {
/*
笔者写的那个太乱了,为了方便观看找了一个清晰一点的
*/
public int findKthPositive(int[] arr, int k) {
int res = 0;
int index = 0;
int ans = 0;
for(int i = 1; i <= arr[arr.length-1]; i++){
//如果相等的话,就找数组的下一个数
if(arr[index] == i){
index++;
//如果不相等,res记录加一个
}else{
res++;
ans = i;
}
//如果相等了就返回
if(res == k)return ans;
}
//如果数组的最后一个数不够,就往后面数
if(res < k)return arr[arr.length-1] + k - res;
return 0;
}
}
给你两个字符串 s 和 t ,你的目标是在 k 次操作以内把字符串 s 转变成 t 。
在第 i 次操作时(1 <= i <= k),你可以选择进行如下操作:
选择字符串 s 中满足 1 <= j <= s.length 且之前未被选过的任意下标 j (下标从 1 开始),并将此位置的字符切换 i 次。
不进行任何操作。
切换 1 次字符的意思是用字母表中该字母的下一个字母替换它(字母表环状接起来,所以 ‘z’ 切换后会变成 ‘a’)。
请记住任意一个下标 j 最多只能被操作 1 次。
如果在不超过 k 次操作内可以把字符串 s 转变成 t ,那么请你返回 true ,否则请你返回 false 。
示例 1:
输入:s = “input”, t = “ouput”, k = 9
输出:true
解释:第 6 次操作时,我们将 ‘i’ 切换 6 次得到 ‘o’ 。第 7 次操作时,我们将 ‘n’ 切换 7 次得到 ‘u’ 。
示例 2:
输入:s = “abc”, t = “bcd”, k = 10
输出:false
解释:我们需要将每个字符切换 1 次才能得到 t 。我们可以在第 1 次操作时将 ‘a’ 切换成 ‘b’ ,但另外 2 个字母在剩余操作中无法再转变为 t 中对应字母。
示例 3:
输入:s = “aab”, t = “bbb”, k = 27
输出:true
解释:第 1 次操作时,我们将第一个 ‘a’ 切换 1 次得到 ‘b’ 。在第 27 次操作时,我们将第二个字母 ‘a’ 切换 27 次得到 ‘b’ 。
提示:
1 <= s.length, t.length <= 10^5
0 <= k <= 10^9
s 和 t 只包含小写英文字母。
class Solution {
public boolean canConvertString(String s, String t, int k) {
if (s.length() != t.length()) {
return false;
}
int[] count = new int[26];
for (int i = 0; i < s.length(); ++i) {
//计算出需要变换的次数
int diff = (t.charAt(i) - s.charAt(i) + 26) % 26;
count[diff]++;
}
for (int i = 1; i < count.length; ++i) {
//计算最大需要变换的次数,如 1 27 53
int change = (i + (count[i] - 1) * 26);
if (change > k) {
return false;
}
}
return true;
}
}
给你一个括号字符串 s ,它只包含字符 ‘(’ 和 ‘)’ 。一个括号字符串被称为平衡的当它满足:
任何左括号 ‘(’ 必须对应两个连续的右括号 ‘))’ 。
左括号 ‘(’ 必须在对应的连续两个右括号 ‘))’ 之前。
比方说 “())”, “())(())))” 和 “(())())))” 都是平衡的, “)()”, “()))” 和 “(()))” 都是不平衡的。
你可以在任意位置插入字符 ‘(’ 和 ‘)’ 使字符串平衡。
请你返回让 s 平衡的最少插入次数。
示例 1:
输入:s = “(()))”
输出:1
解释:第二个左括号有与之匹配的两个右括号,但是第一个左括号只有一个右括号。我们需要在字符串结尾额外增加一个 ‘)’ 使字符串变成平衡字符串 “(())))” 。
示例 2:
输入:s = “())”
输出:0
解释:字符串已经平衡了。
示例 3:
输入:s = “))())(”
输出:3
解释:添加 ‘(’ 去匹配最开头的 ‘))’ ,然后添加 ‘))’ 去匹配最后一个 ‘(’ 。
示例 4:
输入:s = “((((((”
输出:12
解释:添加 12 个 ‘)’ 得到平衡字符串。
示例 5:
输入:s = “)))))))”
输出:5
解释:在字符串开头添加 4 个 ‘(’ 并在结尾添加 1 个 ‘)’ ,字符串变成平衡字符串 “(((())))))))” 。
提示:
1 <= s.length <= 10^5
s 只包含 ‘(’ 和 ‘)’ 。
class Solution {
/*
压栈
左右括号问题(这里是一个左括号对应两个右括号)
*/
public int minInsertions(String s) {
Stack<Character> stack = new Stack<>();
int len = s.length();
int ans = 0;
for (int i = 0; i < len; ++i) {
//如果是左括号就压栈
if (s.charAt(i) == '(') {
stack.push('(');
continue;
}
//后面的都表示当前这一位为)
//如果是最后一位就补全
if (i + 1 == len) {
//如果栈空了,就相当于补一个左括号一个右括号
if (stack.isEmpty())
ans += 2;
//否则就是补一个右括号,出栈一个(
else {
stack.pop();
ans += 1;
}
} else {
//如果下一位也是)就是两个连续的)
if (s.charAt(i + 1) == ')') {
//如果为空,就加一个左括号
if (stack.isEmpty()) {
ans++;
//要不然就直接出栈一个(
} else {
stack.pop();
}
//跳过一位
i++;
//说明下一位是(,也就不是两个连续的)
} else {
//如果栈空了,就相当于补一个左括号一个右括号
if (stack.isEmpty()) {
ans += 2;
//否则就是补一个), 然后出栈一个(然后下一位加上一个( ——这里就相当于不加不减
} else {
ans++;
i++;
}
}
}
}
//加上 栈里面的左括号*2个右括号
ans += stack.size() * 2;
return ans;
}
}
给你一个字符串 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 {
/*
这里要面临的问题就是提取最长的连续一段可以构成的回文
构成回文要知道肯定是每个字母都是 偶数个,这里正好用到^
a^a=0 两个相同的^运算会得到0
运用前缀和,只要从头^到每一位,前面如果有相同的值,就可以算作一个回文,记录一下最长的串
上面说的是偶数的,还可能有一位奇数的字符,我们把0-9都试一遍,如果前面有记录的话,说明有奇数字符构成回文,记录最长长度
都判断完以后,如果前面没有记录过的话,记录当前的前缀和,
Tips:这里使用二进制操作,0==1<<0,1==1<<1,2==1<<2.....9==1<<9这么记录
*/
public int longestAwesome(String s) {
Map<Integer, Integer> map = new HashMap<>();
char[] arr = s.toCharArray();
int temp = 0;
temp ^= (1 << (arr[0]-'0')); // arr[i]记得减'0'
map.put(temp, 0);
map.put(0, -1); // 特殊情况,当前前缀数组可以直接变换成回文串
int ret = 0;
for (int i = 1; i < arr.length; i++) {
temp ^= (1 << (arr[i]-'0'));
if (map.containsKey(temp)) { // 与之前的某一个前缀数组相等,中间这一段可以转换成功
ret = Math.max(ret, i-map.get(temp));
}
for (int j = 0; j < 10; j++) { // 与0-9进行异或后再查看,考虑到奇数长度的回文串
int temp1 = temp ^ (1<<j);
if (map.containsKey(temp1)) {
ret = Math.max(ret, i-map.get(temp1));
}
}
if (!map.containsKey(temp)) { // 不包含当前状态的进行记录
map.put(temp, i);
}
}
if (temp == 0) return arr.length;
return ret;
}
}