java数青蛙_数青蛙[字节跳动专场赛][Java]

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;

}

0fbbd33415e4a35ba50c66c1727ff95b.png

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值