Leetcode第382场周赛

本文涉及的产品
实时计算 Flink 版,5000CU*H 3个月
检索分析服务 Elasticsearch 版,2核4GB开发者规格 1个月
实时数仓Hologres,5000CU*H 100GB 3个月
简介: ```markdown给定字符串`s`,计算按键变更次数,即使用不同键的次数,不考虑大小写差异。例如,`"aAbBcC"`变更了2次。函数`countKeyChanges`实现此功能。另外,求满足特定模式子集最大元素数,`maximumLength`函数使用`TreeMap`统计次数,枚举并构建子集,返回最大长度。最后,Alice和Bob玩鲜花游戏,Alice要赢需满足鲜花总数奇数、顺时针在[1,n]、逆时针在[1,m],返回满足条件的(x, y)对数,可通过奇偶性分类讨论求解。```

一、按键变更的次数

给你一个下标从 0 开始的字符串 s ,该字符串由用户输入。按键变更的定义是:使用与上次使用的按键不同的键。例如 s = "ab" 表示按键变更一次,而 s = "bBBb" 不存在按键变更。

返回用户输入过程中按键变更的次数。

注意:shift 或 caps lock 等修饰键不计入按键变更,也就是说,如果用户先输入字母 'a' 然后输入字母 'A' ,不算作按键变更。

示例 1:

输入:s = "aAbBcC"
输出:2
解释:
从 s[0] = 'a' 到 s[1] = 'A',不存在按键变更,因为不计入 caps lock 或 shift 。
从 s[1] = 'A' 到 s[2] = 'b',按键变更。
从 s[2] = 'b' 到 s[3] = 'B',不存在按键变更,因为不计入 caps lock 或 shift 。
从 s[3] = 'B' 到 s[4] = 'c',按键变更。
从 s[4] = 'c' 到 s[5] = 'C',不存在按键变更,因为不计入 caps lock 或 shift 。
示例 2:

输入:s = "AaAaAaaA"
输出:0
解释: 不存在按键变更,因为这个过程中只按下字母 'a' 和 'A' ,不需要进行按键变更。

解题思路

每个字符都与其自身的上一个字符进行比较,如果不同,则计数+1

代码

public int countKeyChanges(String s) {
   
    int count = 0;
    char prevChar = '\0';

    for (int i = 0; i < s.length(); i++) {
   
        char currentChar = s.charAt(i);
        if(prevChar != '\0' && Character.toLowerCase(currentChar) != Character.toLowerCase(prevChar)){
   
            count++;
        }
        prevChar = s.charAt(i);
    }

    return count;
}

二、子集中元素的最大数量

给你一个 正整数 数组 nums 。

你需要从数组中选出一个满足下述条件的
子集

你可以将选中的元素放置在一个下标从 0 开始的数组中,并使其遵循以下模式:[x, x2, x4, ..., xk/2, xk, xk/2, ..., x4, x2, x](注意,k 可以是任何 非负 的 2 的幂)。例如,[2, 4, 16, 4, 2] 和 [3, 9, 3] 都符合这一模式,而 [2, 4, 8, 4, 2] 则不符合。
返回满足这些条件的子集中,元素数量的 最大值 。

示例 1:

输入:nums = [5,4,1,2,2]
输出:3
解释:选择子集 {4,2,2} ,将其放在数组 [2,4,2] 中,它遵循该模式,且 22 == 4 。因此答案是 3 。
示例 2:

输入:nums = [1,3,2,4]
输出:1
解释:选择子集 {1},将其放在数组 [1] 中,它遵循该模式。因此答案是 1 。注意我们也可以选择子集 {2} 、{4} 或 {3} ,可能存在多个子集都能得到相同的答案。

解题思路

1.使用TreeMap来统计数组nums中每个数字出现的次数。这个数据结构能够根据键值自动排序,方便后续处理。
2.通过枚举的方式,从最小的数字开始,检查每个数字能否作为序列的起点。找到起点之后,获取以其为起点的最大子集,并且对各个不同起点的最大子集长度取最大值,得到最终结果。
3.如何获取最大子集?
通过哈希表存储每个数字出现的次数,并且用num代表应该出现的数字,如果数字为波峰,那么该数字出现的次数为1;如果数字不是波峰,那么该数字出现的次数为2.
4.如果数组的长度为偶数,那么答案为最大长度-1;如果数组的长度为奇数,那么答案为最大长度

代码

public int maximumLength(int[] nums) {
   
     // 1.使用TreeMap统计各数字出现的次数,并利用特性对key从小到大排序
     TreeMap<Integer,Integer> numCntMap = new TreeMap<>();
     for(int num:nums){
   
         numCntMap.put(num, numCntMap.getOrDefault(num, 0)+1);
     }

     // 2.枚举,同时删除已访问的元素
     int maxCnt = 1;
     while(!numCntMap.isEmpty()){
   
         int tmpCnt = 0;
         Map.Entry<Integer,Integer> firstEntry = numCntMap.firstEntry();
         int num = firstEntry.getKey();
         if (num == 1){
   
             // 处理特例1
             maxCnt = Math.max(maxCnt,firstEntry.getValue()%2==0?firstEntry.getValue()-1:firstEntry.getValue());
             numCntMap.pollFirstEntry();
             continue;
         }
         while(numCntMap.containsKey(num)){
   
             int numCnt = numCntMap.get(num);
             numCntMap.remove(num,numCnt);
             // 判断数字出现的次数
             if(numCnt == 1){
   
                 // 波峰,结束枚举
                 tmpCnt++;
                 break;
             } else{
   
                 tmpCnt += 2;
                 num *= num;
             }
         }

         // 处理最后的峰值,峰值元素只取1个
         tmpCnt = tmpCnt % 2 == 0 ? tmpCnt - 1 : tmpCnt;
         maxCnt = Math.max(maxCnt,tmpCnt);
     }
     return maxCnt;
 }

三、Alice和Bob玩鲜花游戏

题目

Alice 和 Bob 在一个长满鲜花的环形草地玩一个回合制游戏。环形的草地上有一些鲜花,Alice 到 Bob 之间顺时针有 x 朵鲜花,逆时针有 y 朵鲜花。

游戏过程如下:

Alice 先行动。
每一次行动中,当前玩家必须选择顺时针或者逆时针,然后在这个方向上摘一朵鲜花。
一次行动结束后,如果所有鲜花都被摘完了,那么 当前 玩家抓住对手并赢得游戏的胜利。
给你两个整数 n 和 m ,你的任务是求出满足以下条件的所有 (x, y) 对:

按照上述规则,Alice 必须赢得游戏。
Alice 顺时针方向上的鲜花数目 x 必须在区间 [1,n] 之间。
Alice 逆时针方向上的鲜花数目 y 必须在区间 [1,m] 之间。
请你返回满足题目描述的数对 (x, y) 的数目。

示例 1:

输入:n = 3, m = 2
输出:3
解释:以下数对满足题目要求:(1,2) ,(3,2) ,(2,1) 。
示例 2:

输入:n = 1, m = 1
输出:0
解释:没有数对满足题目要求。

解题思路

由于Alice先行动,并且要求Alice赢,题目转化为:

  • 鲜花数量x+y是奇数
  • 顺时针方向的鲜花数量x满足1≤x≤n
  • 逆时针方向的鲜花数量y满足1≤y≤m
    由于x+y是奇数,因此xy的奇偶性不同,分类讨论。
    x是奇数且y是偶数时,x的取值个数是floor(n/2)y的取值个数是ceil(m/2)
    x是偶数且y是奇数时,x的取值个数是ceil(n/2)y的取值个数是floor(m/2)
    因此x+y等同于floor(n/2)ceil(m/2)+ceil(n/2)floor(m/2),等同于(n/2)((m+1)/2)+((n+1)/2)(m/2)

    代码

    public long flowerGame(int n, int m) {
         
      return (long) ((n + 1) / 2) * (m / 2) + (long) (n / 2) * ((m + 1) / 2);
    }
    
目录
相关文章
|
6月前
|
Go
golang力扣leetcode 第 291 场周赛
golang力扣leetcode 第 291 场周赛
61 0
|
6月前
|
Go vr&ar
golang力扣leetcode 第 288 场周赛
golang力扣leetcode 第 288 场周赛
52 0
|
6月前
|
Go
golang力扣leetcode 第 286 场周赛
golang力扣leetcode 第 286 场周赛
58 0
|
算法 Android开发
LeetCode 周赛上分之旅 #48 一道简单的树上动态规划问题
学习数据结构与算法的关键在于掌握问题背后的算法思维框架,你的思考越抽象,它能覆盖的问题域就越广,理解难度也更复杂。在这个专栏里,小彭与你分享每场 LeetCode 周赛的解题报告,一起体会上分之旅。
65 1
|
6月前
|
Go
golang力扣leetcode第 294 场周赛
golang力扣leetcode第 294 场周赛
54 0
|
6月前
|
算法 Java Go
golang力扣leetcode 第 293 场周赛
golang力扣leetcode 第 293 场周赛
82 0
|
6月前
|
Go
golang力扣leetcode 第 290 场周赛
golang力扣leetcode 第 290 场周赛
52 0
|
6月前
|
Go C++
golang力扣leetcode 第 284 场周赛
golang力扣leetcode 第 284 场周赛
58 0
|
6月前
|
Go
golang力扣leetcode 第 292 场周赛
golang力扣leetcode 第 292 场周赛
66 0
|
6月前
|
存储
Leetcode第383场周赛
在LeetCode第383场周赛中,选手完成了3道题目。第一题是关于边界上的蚂蚁,蚂蚁根据非零整数数组nums的值移动,返回蚂蚁返回边界上的次数。解题方法是计算数组累加和为0的次数。第二题涉及计算网格的区域平均强度,给定一个灰度图像和阈值,返回每个像素所属区域的平均强度。解题关键在于理解相邻像素和区域定义,并计算平均强度。第三题是恢复单词初始状态的最短时间问题,通过移除前k个字符并添加k个字符,求恢复原词所需的最短时间。解题策略是检查去除前k个字符后的子串是否能作为原词的前缀。
34 1
Leetcode第383场周赛