数青蛙[字节跳动专场赛][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]<num[1] || num[1]<num[2] || num[3]<num[4]){
    			return -1;
    		}
    		// 如果 存在  num里面 每个值 都>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;
    }

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值