LeetCode - 1371 每个元音包含偶数次的最长子字符串(Java JS Python C)

题目来源

1371. 每个元音包含偶数次的最长子字符串 - 力扣(LeetCode)

题目描述

给你一个字符串 s ,请你返回满足以下条件的最长子字符串的长度:每个元音字母,即 'a','e','i','o','u' ,在子字符串中都恰好出现了偶数次。

示例

示例 1

输入:s = "eleetminicoworoep"
输出:13
解释:最长子字符串是 "leetminicowor" ,它包含 e,i,o 各 2 个,以及 0 个 a,u 。


示例 2

输入:s = "leetcodeisgreat"
输出:5
解释:最长子字符串是 "leetc" ,其中包含 2 个 e 。


示例 3

输入:s = "bcbcbc"
输出:6
解释:这个示例中,字符串 "bcbcbc" 本身就是最长的,因为所有的元音 a,e,i,o,u 都出现了 0 次。

提示

  • 1 <= s.length <= 5 x 10^5
  • s 只包含小写英文字母。

题目解析

本题最简单的思路就是双循环暴力枚举所有子串,然后计算子串内各个元音的数目。

但是这种思路肯定会超时。

我们枚举子串的目的,是为了统计子串中各元音字符的数量,而实现该需求的更优思路是利用前缀和。

前缀和的应用场景有非常鲜明的特点,如求解连续范围内的状态,实际例子有:求解任意区间的和。

本题其实也可以当成前缀和问题来看,我们遍历输入串,每遍历一个字符,则对应位置 i 就有一个前缀状态 preSum[i],本题preSum[i] 表示 [0, i] 范围内各个元音字符的数量,具体表现为:

preSum[i] = {
    ‘a’:aCount,

    'e':eCount,

    'i':iCount,

    'o':oCount,

    'u':uCount
}

那么,如果我们要求解范围[i, j]子串的各个元音的数量,即可通过 preSum[j] - preSum[i-1] 得到。
更多前缀和知识请看:算法设计 - 前缀和 & 差分数列_算法设计 - 前缀和 & 差分数列_伏城之外的博客-csdn博客-CSDN博客

但是光靠前缀和,我们还是要枚举所有子串,依旧会超时。

本题要求我们求解最长的子串,子串范围内各个元音数量为偶数。我们假设:

preSum[j] 位置各个元音的数量分别为:aCount偶数,eCount奇数,iCount偶数,oCount偶数,uCount奇数

再假设 [i, j] 范围内各个元音的数量都为偶数,那么此时preSum[i-1]的各个元音的数量应该是多少呢?

答:必然和preSum[j] 对应元音的数量同奇偶性。因为:

  • 奇数 - 奇数 = 偶数
  • 偶数 - 偶数 = 偶数

比如 preSum[j] = {aCount: 8,  eCount: 3, iCount: 6, oCount: 4, uCount: 5};

且  preSum[i-1] = {aCount: 2,  eCount: 1, iCount: 4, oCount: 2, uCount: 3};

那么 [i, j] 范围内,各个元音的数量为:

  • aCount = 8 - 2 = 6
  • eCount = 3 - 1 = 2
  • iCount = 6 - 4 = 2
  • oCount = 4 - 2 = 2
  • uCount = 5 - 3 = 2

因此,当我们得到 preSum[j] 后,我们应该在 i - 1 ∈ [0, j - 1] 范围内找到一个 preSum[i-1] 和 preSum[j] 的各个元音同奇偶性的,且 i - 1要最小,这样得到 [i, j] 范围子串才是 [0, j] 范围内一个最长的且各个元音数量都为偶数的子串。

此时,逻辑虽然得到了优化,但是我们依旧要遍历 0 ~ j - 1 范围内的位置 i,且需要对比对应 preSum[i] 和 preSum[j] 的各个元音的奇偶性,这样依然会超时。

接下来要用到状态压缩了。

由于我们只关注各个元音的数量的奇偶性,即每个元音的数量要么为奇数,要么为偶数,假设我们用0表示偶数,用1表示奇数的话。

那么初始 preSum[0] 的各个元音的状态就可以表示为二进制数:00000

各个二进制位和对应元音对应,如上图所示,当二进制位值为0时,表示对应元音数量为偶数个,当二进制位值为1时,表示对应元音数量为奇数个。

这样我们就完成了 preSum[i] 的状态压缩。

那么状态压缩后的preSum[j] 和 preSum[j+1] 如何进行前缀和累进呢?

假设 j+1 位置的字符是 'e',那么代表,preSum[j] 的二进制数 中 'e' 对应的位的性质反转,即奇变偶,偶变奇。

比如 preSum[j] = 01010,而 s[j+1] 字符是 'e',则 preSum[j+1] = 00010

此时我们完全可以用异或运算从preSum[j]得出preSum[j+1]的结果,比如 s[j+1] == 'e',则

preSum[j + 1] = preSum[j] ^ 01000

其中 01000 代表是新增一个'e'字符,同理

  • 10000 代表新增一个 'a' 字符
  • 01000 代表新增一个 'e' 字符
  • 00100 代表新增一个 'i' 字符
  • 00010 代表新增一个 'o' 字符
  • 00001 代表新增一个 'u' 字符

接下来就是,比较preSum[i] 和 preSum[j] 的各个元音的奇偶性是否一致,就可以直接将对应二进制数进行值比较即可,值相同,则奇偶性一致,否则不一致。

比如:preSum[i-1] = 01010,preSum[j] = 01010,那么二者的各个元音的奇偶性就一致。

当我们完成preSum[i]的状态压缩后,我们就可以定义一个哈希表map来记录某个压缩状态最早出现的位置,map的key时压缩状态,val时该压缩状态的最早出现位置。

即:我们求解[0, j]范围前缀子串的压缩状态status = preSum[j]后:

  • 如果map存在key=status,那么status状态最早出现位置为 map[staus],我们定义 i = map[status],那么i,j位置的前缀子串内部的各个元音的数量是同奇偶性的,即 [i+1, j] 范围内子串的各个元音的数量B必然都是偶数,[i+1, j] 范围子串是一个符合要求的子串,我们需要记录该子串的长度 j - (i + 1) + 1 = j - i
  • 如果 map 不存在 key == status,那么status状态最早出现的位置就是 j,我们需要记录 map[status] = j

按照逻辑,我们只要遍历一遍字符串s,即可找到最长的目标子串。

JS算法源码

/**
 * @param {string} s
 * @return {number}
 */
var findTheLongestSubstring = function (s) {
  // "前缀子串"中各个元音的奇偶状态
  // 00000
  // aeiou
  // 元音字母和二进制位的对应关系如上,如果二进制位值位0,代表对应元音字符数量有偶数个,如果二进制位值为1,代表对应元音字符数量有奇数个
  // 初始未遍历时,没有子串,此时各个元音的数量都为0,即偶数个,因此所有二进制位值位0
  let status = 0b00000;

  // map记录某个状态的最早出现位置
  // 压缩状态用五位二进制数表示,因此最多有32种状态
  const map = new Array(32).fill(-2); // -2是一个不可能的位置,即初始时所有状态都未出现过
  // 00000 状态对应的十进制数为0,最早出现位置是-1,即未遍历,没有子串时
  map[0] = -1;

  // 记录最长的符合要求的子串长度
  let maxLen = 0;

  for (let i = 0; i < s.length; i++) {
    // 如果遍历的字符s[i]是元音字母,则变更对应二进制位的奇偶性
    switch (s[i]) {
      case "a":
        status ^= 0b10000;
        break;
      case "e":
        status ^= 0b01000;
        break;
      case "i":
        status ^= 0b00100;
        break;
      case "o":
        status ^= 0b00010;
        break;
      case "u":
        status ^= 0b00001;
        break;
    }

    // 如果对应状态的最早出现位置为-2,表示没有出现过对应状态,否则map[status]即status状态最早出现的位置
    if (map[status] != -2) {
      // 当前位置 i 的状态为status,而最早出现status状态的位置是 map[status],两个位置同奇偶性,因此他们形成的范围内子串是符合要求的
      maxLen = Math.max(maxLen, i - map[status]);
    } else {
      // 如果对应状态之前未出现过,则当前位置 i 就是该状态的最早出现位置
      map[status] = i;
    }
  }

  return maxLen;
};

Java算法源码

import java.util.Arrays;

class Solution {
  public int findTheLongestSubstring(String s) {
    // "前缀子串"中各个元音的奇偶状态
    // 00000
    // aeiou
    // 元音字母和二进制位的对应关系如上,如果二进制位值位0,代表对应元音字符数量有偶数个,如果二进制位值为1,代表对应元音字符数量有奇数个
    // 初始未遍历时,没有子串,此时各个元音的数量都为0,即偶数个,因此所有二进制位值位0
    int status = 0b00000;

    // map记录某个状态的最早出现位置
    // 压缩状态用五位二进制数表示,因此最多有32种状态
    int[] map = new int[32]; // -2是一个不可能的位置
    Arrays.fill(map, -2);

    // 00000 状态对应的十进制数为0,最早出现位置是-1,即未遍历,没有子串时
    map[0] = -1;

    // 记录最长的符合要求的子串长度
    int maxLen = 0;

    for (int i = 0; i < s.length(); i++) {
      // 如果遍历的字符s[i]是元音字母,则变更对应二进制位的奇偶性
      switch (s.charAt(i)) {
        case 'a':
          status ^= 0b10000;
          break;
        case 'e':
          status ^= 0b01000;
          break;
        case 'i':
          status ^= 0b00100;
          break;
        case 'o':
          status ^= 0b00010;
          break;
        case 'u':
          status ^= 0b00001;
          break;
      }

      // 如果对应状态的最早出现位置为-2,表示没有出现过对应状态,否则map[status]即status状态最早出现的位置
      if (map[status] != -2) {
        // 当前位置 i 的状态为status,而最早出现status状态的位置是 map[status],两个位置同奇偶性,因此他们形成的范围内子串是符合要求的
        maxLen = Math.max(maxLen, i - map[status]);
      } else {
        // 如果对应状态之前未出现过,则当前位置 i 就是该状态的最早出现位置
        map[status] = i;
      }
    }

    return maxLen;
  }
}

Python算法源码

class Solution(object):
    def findTheLongestSubstring(self, s):
        """
        :type s: str
        :rtype: int
        """
        # "前缀子串"中各个元音的奇偶状态
        # 00000
        # aeiou
        # 元音字母和二进制位的对应关系如上,如果二进制位值位0,代表对应元音字符数量有偶数个,如果二进制位值为1,代表对应元音字符数量有奇数个
        # 初始未遍历时,没有子串,此时各个元音的数量都为0,即偶数个,因此所有二进制位值位0
        status = 0b00000

        # map记录某个状态的最早出现位置, -2是一个不可能的位置, 即初始时各个状态都没有出现过
        # 压缩状态用五位二进制数表示,因此最多有32种状态
        map = [-2] * 32
        # 00000 状态对应的十进制数为0,最早出现位置是-1,即未遍历,没有子串时
        map[0] = -1

        # 记录最长的符合要求的子串长度
        maxLen = 0

        for i in range(len(s)):
            c = s[i]

            # 如果遍历的字符s[i]是元音字母,则变更对应二进制位的奇偶性
            if c == 'a':
                status ^= 0b10000
            elif c == 'e':
                status ^= 0b01000
            elif c == 'i':
                status ^= 0b00100
            elif c == 'o':
                status ^= 0b00010
            elif c == 'u':
                status ^= 0b00001

            # 如果对应状态的最早出现位置为-2,表示没有出现过对应状态,否则map[status]即status状态最早出现的位置
            if map[status] != -2:
                # 当前位置 i 的状态为status,而最早出现status状态的位置是 map[status],两个位置同奇偶性,因此他们形成的范围内子串是符合要求的
                maxLen = max(maxLen, i - map[status])
            else:
                # 如果对应状态之前未出现过,则当前位置 i 就是该状态的最早出现位置
                map[status] = i

        return maxLen

C算法源码

int findTheLongestSubstring(char* s) {
    // "前缀子串"中各个元音的奇偶状态
    // 00000
    // aeiou
    // 元音字母和二进制位的对应关系如上,如果二进制位值位0,代表对应元音字符数量有偶数个,如果二进制位值为1,代表对应元音字符数量有奇数个
    // 初始未遍历时,没有子串,此时各个元音的数量都为0,即偶数个,因此所有二进制位值位0
    int status = 0b00000;

    // map记录某个状态的最早出现位置
    // 压缩状态用五位二进制数表示,因此最多有32种状态
    int map[32];
    for(int i=0; i<32; i++) map[i] = -2;  // -2是一个不可能的位置,即初始时所有状态都未出现过

    // 00000 状态对应的十进制数为0,最早出现位置是-1,即未遍历,没有子串时
    map[0] = -1;

    // 记录最长的符合要求的子串长度
    int maxLen = 0;

    int i=0;
    while(s[i] != '\0') {
        char c = s[i];

        // 如果遍历的字符s[i]是元音字母,则变更对应二进制位的奇偶性
        switch(c) {
            case 'a':
                status ^= 0b10000;
                break;
            case 'e':
                status ^= 0b01000;
                break;
            case 'i':
                status ^= 0b00100;
                break;
            case 'o':
                status ^= 0b00010;
                break;
            case 'u':
                status ^= 0b00001;
                break;
        }

        // 如果对应状态的最早出现位置为-2,表示没有出现过对应状态,否则map[status]即status状态最早出现的位置
        if (map[status] != -2) {
            // 当前位置 i 的状态为status,而最早出现status状态的位置是 map[status],两个位置同奇偶性,因此他们形成的范围内子串是符合要求的
            maxLen = (int) fmax(maxLen, i - map[status]);
        } else {
            // 如果对应状态之前未出现过,则当前位置 i 就是该状态的最早出现位置
            map[status] = i;
        }

        i++;
    }

    return maxLen;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/315642.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

DrGraph原理示教 - OpenCV 4 功能 - 边界填充

今天简单来看一下OpenCV中的边界填充 param src Source image. param dst Destination image of the same type as src and the size Size(src.colsleftright, src.rowstopbottom) . param top the top pixels param bottom the bottom pixels param left the left pixels par…

Redis-浅谈redis.conf配置文件

Redis.conf Redis.conf是Redis的配置文件&#xff0c;它包含了一系列用于配置Redis服务器行为和功能的选项。 以下是Redis.conf中常见的一些选项配置&#xff1a; bind: 指定Redis服务器监听的IP地址&#xff0c;默认为127.0.0.1&#xff0c;表示只能本地访问&#xff0c;可以…

少儿编程 2023年12月电子学会图形化编程等级考试Scratch二级真题解析(判断题)

2023年12月scratch编程等级考试二级真题 判断题(共10题,每题2分,共20分) 26、声音Medieval1的长度是9.68秒,运行下列程序1或程序2都能实现,播放声音2秒后,声音停止角色移动100步 答案:对 考点分析:考查积木综合使用,重点考查声音积木的使用 程序1中用的是等待播完…

暴打小苹果

欢迎来到程序小院 暴打小苹果 玩法&#xff1a;鼠标左键点击任意区域可发招暴打&#xff0c;在苹果到达圆圈时点击更容易击中&#xff0c; 30秒挑战暴打小苹果&#xff0c;打中一次20分&#xff0c;快去暴打小苹果吧^^。开始游戏https://www.ormcc.com/play/gameStart/247 htm…

nova组件讲解和glance对接swift

1、openstack架构 &#xff08;1&#xff09;openstack是一种SOA架构&#xff08;微服务就是从这种架构中剥离出来的&#xff09; &#xff08;2&#xff09;这种SOA架构&#xff0c;就是把每个服务独立成一个组件&#xff0c;每个组件通过定义好的api接口进行互通 &#xff…

如何优雅的只在当前页面中覆盖ui库中组件的样式(vue的问题)

首先我们vue文件的样式都是写在<style lang"less" scoped></style>标签中的&#xff0c;加scoped是为了使得样式只在当前页面有效。那么问题来了&#xff0c;看图&#xff1a; 我们正常写的所有样式&#xff0c;都会被加上[data-v-23d425f8]这个属性&…

【大厂秘籍】 - Redis持久化篇

创作不易&#xff0c;你的关注分享就是博主更新的最大动力&#xff0c; 每周持续更新 微信搜索【 企鹅君】关注还能领取学习资料喔&#xff0c;第一时间阅读(比博客早两到三篇) 求关注❤️ 求点赞❤️ 求分享❤️ 对博主真的非常重要 企鹅君原创&#xff5c;GitHub开源项目gith…

UL2034详细介绍UL 安全单站和多站一氧化碳报警器标准

在介绍相关标准之前先介绍一下UL认证和UL测试报告的区别&#xff0c;检测认证行业6年老司机 UL认证是自愿性的认证&#xff0c;需要检测产品和审核工厂&#xff0c;每个季度审核一次&#xff0c;费用高、时间久&#xff0c;而且审厂非常的严格。 UL测试报告是根据产品选用相应…

Linux中安装字体

问题说明 wps 安装后打开文件部分字体出现乱码&#xff0c;原因主要是linux中缺少windows中的相关字体&#xff0c;只要从windows电脑中的字体拷贝到linux系统中并安装就能解决问题 对ubuntu 和manjora有效。 安装字体 字体下载地址可参考附录 在 Linux 中&#xff0c;一次…

传奇手游详细图文架设教程

开始架设 1. 架设条件 传世手游架设需要准备&#xff1a; linux 服务器&#xff0c;建议 CentOs 7.6 版本&#xff0c;游戏源码&#xff0c; 游戏运行大约占 2.5G 左右内存。 2. 安装宝塔及环境 宝塔是一个服务器运维管理软件&#xff0c;安装命令&#xff1a; yum inst…

掌握 Vue 响应式系统,让数据驱动视图(上)

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

Django框架完成读者浏览书籍,图书详情页,借阅管理

前情回顾&#xff1a; 使用Django框架实现简单的图书借阅系统——完成图书信息管理 文章目录 1.完成展示图书信息功能1.1django 静态资源管理问题1.2编写图书展示模板HTML 2.完成图书详情页功能2.1从后端获取图书详情信息2.2详情页面展示图书数据 3.完成借阅管理功能3.1管理员…

QT上位机开发(文本编辑器的界面开发)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 文本编辑器是编程开发中经常使用到的一个软件&#xff0c;比如说notepad就是其中一种。这里说编写一个文本编辑器&#xff0c;并不是说真的要写一个…

linux 内存

linux内存分类 按用途分 stack heap(brk,sbrk , mmap), 文件映射&#xff0c; bss&#xff0c; data , text, 还有page cache&#xff0c; slab&#xff08;kmalloc连续&#xff09;, vmalloc等内核深处的。 属性 进程OOM 对于进程来说&#xff0c;堆泄漏在死亡时是没问题 但…

【Java SE语法篇】7.面向对象——类和对象

&#x1f4da;博客主页&#xff1a;爱敲代码的小杨. ✨专栏&#xff1a;《Java SE语法》 ❤️感谢大家点赞&#x1f44d;&#x1f3fb;收藏⭐评论✍&#x1f3fb;&#xff0c;您的三连就是我持续更新的动力❤️ 文章目录 1. 面向对象程序设计概述1.1 类1.2 对象1.3 类之间的…

UE5 实现RPG游戏操作控制

在UE5以后&#xff0c;epic抛弃了之前的那一套操作输入系统&#xff0c;使用了一套新的增强输入作为替代&#xff0c;目的主要是解决经常切换操作时的问题&#xff08;操作人物上车以后&#xff0c;可以直接切换成操作汽车的一套输入&#xff09;接下来&#xff0c;将实现如何使…

用React给XXL-JOB开发一个新皮肤(三):实现登录页和Layout骨架

目录 一. 简述二. 接口服务调整 2.1. 登录接口2.2. 登出接口2.3. 修改密码接口2.4. 修改配置文件 三. 前端HTTP 请求四. 登录页面 4.1. 搭建登录页面4.2. 对接登录接口 五. Layout 骨架 5.1. 搭建骨架5.2. Header5.3. 修改密码5.4. 退出登录 六. 其他 一. 简述 上一篇文章我…

Android代码混淆

Android之代码混淆 代码混淆的作用设置混淆1. 在模块目录下的 build.gradle 文件中配置以下代码2. 在 proguard-rules.pro 文件中添加混淆规则 通用混淆规则常用匹配符常用命令注意事项如何查看是否已混淆 代码混淆的作用 1.令 APK 难以被逆向工程&#xff0c;即很大程度上增加…

Nightingale 夜莺监控系统 - 监控篇(2)

Author&#xff1a;rab 官方文档&#xff1a;https://flashcat.cloud/docs/content/flashcat-monitor/categraf/3-configuration/ 目录 前言一、Categraf 配置文件二、Input 插件配置文件2.1 插件说明2.2 通用配置2.2.1 配置采集频率 interval2.2.2 配置采集实例 instances2.2…

Spring Boot - Application Events 的发布顺序_ContextRefreshedListener

文章目录 Pre概述Code源码分析 Pre Spring Boot - Application Events 的发布顺序_ApplicationEnvironmentPreparedEvent 概述 Spring Boot 的广播机制是基于观察者模式实现的&#xff0c;它允许在 Spring 应用程序中发布和监听事件。这种机制的主要目的是为了实现解耦&#…