5390. 数青蛙
给你一个字符串 croakOfFrogs,它表示不同青蛙发出的蛙鸣声(字符串 "croak" )的组合。由于同一时间可以有多只青蛙呱呱作响,所以 croakOfFrogs 中会混合多个 “croak” 。请你返回模拟字符串中所有蛙鸣所需不同青蛙的最少数目。
注意:要想发出蛙鸣 "croak",青蛙必须 依序 输出 ‘c’, ’r’, ’o’, ’a’, ’k’ 这 5 个字母。如果没有输出全部五个字母,那么它就不会发出声音。
如果字符串 croakOfFrogs 不是由若干有效的 "croak" 字符混合而成,请返回 -1 。
示例 1:
输入:croakOfFrogs = "croakcroak"
输出:1
解释:一只青蛙 “呱呱” 两次
示例 2:
输入:croakOfFrogs = "crcoakroak"
输出:2
解释:最少需要两只青蛙,“呱呱” 声用黑体标注
第一只青蛙 "crcoakroak"
第二只青蛙 "crcoakroak"
示例 3:
输入:croakOfFrogs = "croakcrook"
输出:-1
解释:给出的字符串不是 "croak" 的有效组合。
示例 4:
输入:croakOfFrogs = "croakcroa"
输出:-1
提示:
1 <= croakOfFrogs.length <= 10^5
字符串中的字符只有 'c', 'r', 'o', 'a' 或者 'k'
思路:仔细想想,数据量最大10^5能采取的方法,比较直观上讲有O(n)一次遍历,O(n)log(n)方法,O(n)log(n)怎么下手,没法下手。所以这里也比较容易想,只能O(n)并且,题目中也说了是需要次序遍历的。
那么怎么设计算法呢?
题目求任意时刻最少需要需要的青蛙的个数,已知题目中青蛙可以任意时刻叫的,就是有穿插的意思。所以,可能有很多种组合,那么我们就可以记录‘c’, ’r’, ’o’, ’a’, ’k’ 这五个字符的次数,初始都为0,
又因为如果字符串 croakOfFrogs 不是由若干有效的 "croak" 字符混合而成,请返回 -1 。也就是说 比较按顺序
叫出‘c’, ’r’, ’o’, ’a’, ’k’这五个字符,所以数量上是严格>=的
如果不符这种要求那么就是返回-1另外还需要提前判断一种情况就是如果长度%5!=0也是返回-1
因为最后可能会出现1,1,1,1,0的情况,这也是不符题意的,应该返回-1,因为叫不出来。
public int minNumberOfFrogs(String croakOfFrogs) {
int[] num = new int[5];// 分别记录‘c’, ’r’, ’o’, ’a’, ’k’的个数
int len = croakOfFrogs.length();
int res = 0;
// 特判
if(len %5 !=0)return -1;
for(int i = 0 ; i < len ; i++){
if(croakOfFrogs.charAt(i)=='c'){
num[0]++;
}else if(croakOfFrogs.charAt(i)=='r'){
num[1]++;
}else if(croakOfFrogs.charAt(i)=='o'){
num[2]++;
}else if(croakOfFrogs.charAt(i)=='a'){
num[3]++;
}else if(croakOfFrogs.charAt(i)=='k'){
num[4]++;
}
// 任意时刻,num数组都是 递减的,如果 不符这一情况,肯定不是有效的。
// 因为要满足有效的蛙叫中间可能有很多个再叫,而且是 必须从头叫。
if(num[0]1 那么说明 有只青蛙 交完了,那么就需要 全部-1
// 维护一个当前num数组中,的一个最大值,即可, 即最少需要的青蛙个数。
if(num[0]>=1 && num[1]>=1 && num[2]>=1 && num[3]>=1 && num[4]>=1){
num[0]--;
num[1]--;
num[2]--;
num[3]--;
num[4]--;
}
for(int j = 0 ; j < 5 ; j ++){
res = Math.max(res, num[j]);
}
}
return res;
}