Java实现LeetCode第32场双周赛(题号5468,5469,5470,5485)

有什么不对的地方欢迎大佬评论区评论

5468. 第 k 个缺失的正整数(遍历)
5469. K 次操作转变字符串(遍历记录)
5470. 平衡括号字符串的最少插入次数(压栈)
5485. 找出最长的超赞子字符串(位运算,前缀和)

5468. 第 k 个缺失的正整数

给你一个 严格升序排列 的正整数数组 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;
    }
}

5469. K 次操作转变字符串

给你两个字符串 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;
    }
}

5470. 平衡括号字符串的最少插入次数

给你一个括号字符串 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;
    }
}

5485. 找出最长的超赞子字符串

给你一个字符串 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;
    }
 
}

你可能感兴趣的:(LeetCode双周赛,Java,LeetCode,双周赛,32)