力扣(leetcode)每日一题 2516 每种字符至少取 K 个 | 滑动窗口

2516. 每种字符至少取 K 个

给你一个由字符 'a''b''c' 组成的字符串 s 和一个非负整数 k 。每分钟,你可以选择取走 s 最左侧 还是 最右侧 的那个字符。

你必须取走每种字符 至少 k 个,返回需要的 最少 分钟数;如果无法取到,则返回 -1

示例 1:

**输入:**s = "aabaaaacaabc", k = 2

**输出:**8
解释:

从 s 的左侧取三个字符,现在共取到两个字符 'a' 、一个字符 'b' 。

从 s 的右侧取五个字符,现在共取到四个字符 'a' 、两个字符 'b' 和两个字符 'c' 。

共需要 3 + 5 = 8 分钟。

可以证明需要的最少分钟数是 8 。

示例 2:

**输入:**s = "a", k = 1
输出: -1

**解释:**无法取到一个字符 'b' 或者 'c',所以返回 -1 。

提示:

  • 1 <= s.length <= 105
  • s 仅由字母 'a''b''c' 组成
  • 0 <= k <= s.length
题解

这里可以想到滑动窗口的变种。我左边一个指针,右边一个指针。左边的先顶到满足的最右边,然后右边顶向左边的同时,左边的指针可以弹出来

这个过程中不断刷新最小值

java 复制代码
public static int takeCharacters(String s, int k) {  
    // 变相滑动窗口  
    int[] count = new int[3];  
    char[] charArray = s.toCharArray();  
    int index = 0;  
    int length = charArray.length;  
    for (int i = 0; i < length; i++) {  
        if (!verify(count, k)) {  
            if (charArray[i] == 'a') {  
                count[0]++;  
            } else if (charArray[i] == 'b') {  
                count[1]++;  
            } else if (charArray[i] == 'c') {  
                count[2]++;  
            }  
            index++;  
        } else {  
            break;  
        }  
    }  
    if (!verify(count, k)) {  
        return -1;  
    }  
    int res = index;// 个数  
    int rightIndex = 0;  
    for (int i = length - 1; i > 0 && index > 0 && rightIndex < res; i--) {  
        rightIndex++;  
        if (charArray[i] == 'a') {  
            count[0]++;  
        } else if (charArray[i] == 'b') {  
            count[1]++;  
        } else if (charArray[i] == 'c') {  
            count[2]++;  
        }  
        // 先加上最后面一位,然后弹出index的位数  
        while (verify(count, k) && index > 0) {  
            index--;  
            if (charArray[index] == 'a') {  
                count[0]--;  
            } else if (charArray[index] == 'b') {  
                count[1]--;  
            } else if (charArray[index] == 'c') {  
                count[2]--;  
            }  
  
        }  
        // 如果还是满足,就不需要加一  
        if (verify(count, k)) {  
            res = Math.min(res, index + rightIndex);  
        } else {  
            res = Math.min(res, index + rightIndex + 1);  
        }  
  
    }  
    return res;  
}  
  
public static boolean verify(int[] count, int k) {  
    return count[0] >= k && count[1] >= k && count[2] >= k;  
}

这里index就是左边的个数,leftIndex就是右边的个数。单独命名代码好写点。

退出语句皆是下 i > 0 && index > 0 && rightIndex < res i一定要大于0不说了,index如果是0,说明左指针已经弹完了,就可以退出了,还有如果右指针弹入的数量比刷新的值还大,也没有继续的意义了

这里的if是可以优化的

java 复制代码
  
public static int takeCharacters2(String s, int k) {  
    // 变相滑动窗口  
    int[] count = new int[3];  
    char[] charArray = s.toCharArray();  
    int index = 0;  
    int length = charArray.length;  
    for (int i = 0; i < length; i++) {  
        if (!verify(count, k)) {  
            count[charArray[i] - 'a']++;  
            index++;  
        } else {  
            //index--;// 到了4更新  然后来到5发现成功了,其实index在4的时候已经成功了 这里减去1  
            break;  
        }  
    }  
    if (!verify(count, k)) {  
        return -1;  
    }  
    int res = index;// 个数  
    int rightIndex = 0;  
    for (int i = length - 1; i > 0 && index > 0 && rightIndex < res; i--) {  
        rightIndex++;  
        count[charArray[i] - 'a']++;  
        // 先加上最后面一位,然后弹出index的位数  
        while (verify(count, k) && index > 0) {  
            index--;  
            count[charArray[index] - 'a']--;  
        }  
        // 如果还是满足,就不需要加一  
        if (verify(count, k)) {  
            res = Math.min(res, index + rightIndex);  
        } else {  
            res = Math.min(res, index + rightIndex + 1);  
        }  
    }  
    return res;  
}

  
public static boolean verify(int[] count, int k) {  
    return count[0] >= k && count[1] >= k && count[2] >= k;  
}  
总结

其实滑动窗口的代码并不好写,涉及到临界的判断。如果是第一次写,很可能会自闭。我已经写过很多次了,还是不能一笔写完

这里花了这么多时间是想着把index和rightIndex融入到for循环。

然后这个忘记写了

java 复制代码
 // 如果还是满足,就不需要加一  
        if (verify(count, k)) {  
            res = Math.min(res, index + rightIndex);  
        } else {  
            res = Math.min(res, index + rightIndex + 1);  
        }  

这不是最优解,但是最近比较忙,就不研究最优解了

相关推荐
jiao_mrswang44 分钟前
leetcode-18-四数之和
算法·leetcode·职场和发展
qystca1 小时前
洛谷 B3637 最长上升子序列 C语言 记忆化搜索->‘正序‘dp
c语言·开发语言·算法
薯条不要番茄酱1 小时前
数据结构-8.Java. 七大排序算法(中篇)
java·开发语言·数据结构·后端·算法·排序算法·intellij-idea
今天吃饺子1 小时前
2024年SCI一区最新改进优化算法——四参数自适应生长优化器,MATLAB代码免费获取...
开发语言·算法·matlab
是阿建吖!1 小时前
【优选算法】二分查找
c++·算法
王燕龙(大卫)1 小时前
leetcode 数组中第k个最大元素
算法·leetcode
不去幼儿园2 小时前
【MARL】深入理解多智能体近端策略优化(MAPPO)算法与调参
人工智能·python·算法·机器学习·强化学习
Mr_Xuhhh2 小时前
重生之我在学环境变量
linux·运维·服务器·前端·chrome·算法
盼海3 小时前
排序算法(五)--归并排序
数据结构·算法·排序算法
网易独家音乐人Mike Zhou6 小时前
【卡尔曼滤波】数据预测Prediction观测器的理论推导及应用 C语言、Python实现(Kalman Filter)
c语言·python·单片机·物联网·算法·嵌入式·iot