LeetCode 周赛上分之旅 #33 摩尔投票派上用场

news2024/11/26 8:49:23

⭐️ 本文已收录到 AndroidFamily,技术和职场问题,请关注公众号 [彭旭锐] 和 [BaguTree Pro] 知识星球提问。

学习数据结构与算法的关键在于掌握问题背后的算法思维框架,你的思考越抽象,它能覆盖的问题域就越广,理解难度也更复杂。在这个专栏里,小彭与你分享每场 LeetCode 周赛的解题报告,一起体会上分之旅。

本文是 LeetCode 上分之旅系列的第 33 篇文章,往期回顾请移步到文章末尾~

周赛 354

T1. 特殊元素平方和(Easy)

  • 标签:模拟、数学

T2. 数组的最大美丽值(Medium)

  • 标签:排序、二分查找、同向双指针

T3. 合法分割的最小下标(Medium)

  • 标签:数学、前后缀分解

T4. 最长合法子字符串的长度(Hard)

  • 标签:同向双指针


T1. 特殊元素平方和(Easy)

https://leetcode.cn/problems/sum-of-squares-of-special-elements/

题解一(模拟)

简单模拟题,枚举每个下标检查是否能被 n 整除,同时记录结果。

class Solution {
public:
    int sumOfSquares(vector<int>& nums) {
        int ret = 0;
        int n = nums.size();
        for (int i = 0; i < nums.size(); i++) {
            if (n % (i + 1) == 0) ret += nums[i] * nums[i];
        }
        return ret;
    }
};

复杂度分析:

  • 时间复杂度: O ( n ) O(n) O(n)
  • 空间复杂度: O ( 1 ) O(1) O(1)

题解二(模拟 + 优化)

事实上,当下标 i 可以被 n 整除时,那么有下标 n / i 也可以被 n 整除,因此我们只需要检查 [0, \sqrt(n)] 的范围。

  • 1、将 nums[0] 和 nums[n - 1] 的平方值添加到结果中(如果数组长度不大于 1,则不需要添加 nums[n - 1] 的影响);
  • 2、从 2 到 sqrt(n) 的范围内遍历所有元素下标 i,如果 n 能够被 i 整除,那么我们将 nums[i-1] 的平方值和 nums[n/i-1] 的平方值分别添加到结果中(如果 i 和 n/i 相等,我们只添加其中一个值,以避免重复);
class Solution {
public:
    int sumOfSquares(vector<int>& nums) {
        int ret = nums[0] * nums[0];
        int n = nums.size();
        if (n < 2) return ret;
        ret += nums[n - 1] * nums[n - 1];
        for (int i = 2; i <= sqrt(n); i++) {
            if (n % i != 0) continue;
            ret += nums[i - 1] * nums[i - 1];
            if (i != n / i) {
                ret += nums[n / i - 1] * nums[n / i - 1];
            }
        }
        return ret;
    }
};

复杂度分析:

  • 时间复杂度: O ( ( n ) ) O(\sqrt(n)) O(( n))
  • 空间复杂度: O ( 1 ) O(1) O(1)

其他语言解法见 LeetCode 题解页:枚举优化的 O(sqrt(n) 时间解法(C++/Python/Kotlin)


T2. 数组的最大美丽值(Medium)

https://leetcode.cn/problems/maximum-beauty-of-an-array-after-applying-operation/

题解一(排序 + 二分查找)

根据题目操作描述,每个元素都可以修改为范围在 [nums[i] - k, nums[i] + k] 之间的任意元素,我们把两个元素的差视为元素的相似度,那么差值小于 2*k 的两个数就能够转换为相等数(增大较小数,同时减小较大数)。

由于美丽值和数组顺序无关,我们先对数组排序,然后枚举元素作为左值,再寻找最远可匹配的右值(nums[i] + 2 * k),可以使用二分查找寻找不大于右值的最大元素。

class Solution {
public:
    int maximumBeauty(vector<int>& nums, int k) {
        sort(nums.begin(), nums.end());
        int ret = 0;
        for (int i = 0; i < nums.size(); i++) {
            int left = i;
            int right = nums.size() - 1;
            while (left < right) {
                int mid = (left + right + 1) / 2;
                if (nums[mid] > nums[i] + 2 * k) {
                    right = mid - 1;
                } else {
                    left = mid;
                }
            }
            ret = max(ret, left - i + 1);
        }
        return ret;
    }
};

复杂度分析:

  • 时间复杂度: O ( n l g n ) O(nlgn) O(nlgn) 瓶颈在排序,模拟时间为 O ( n l g n ) O(nlgn) O(nlgn)
  • 空间复杂度: O ( l g n ) O(lgn) O(lgn) 瓶颈在排序。

题解二(排序 + 同向双指针)

根据题目操作描述,每个元素都可以修改为范围在 [nums[i] - k, nums[i] + k] 之间的任意元素,我们把这个范围视为一个可选区间。那么问题的最大美丽值正好就是所有区间的最多重叠数,这就是经典的 leetcode 253. 会议室 II 问题

由于区间重叠数和顺序无关,我们可以对所有元素排序(由于区间长度相等,等价于按照结束时间排序),使用同向双指针求解:

  • 维护重叠区间的左右指针 i 和 j
  • 如果当前区间 [j] 与左指针指向的区间不重叠,则将左指针 i 向右移动,并记录最大重叠数
class Solution {
public:
    int maximumBeauty(vector<int>& nums, int k) {
        sort(nums.begin(), nums.end());
        int i = 0;
        int ret = 0;
        for (int j = 0; j < nums.size(); j++) {
            while (nums[j] - k > nums[i] + k) i++;
            ret = max(ret, j - i + 1);
        }
        return ret;
    }
};

复杂度分析:

  • 时间复杂度: O ( n l g n ) O(nlgn) O(nlgn) 瓶颈在排序,同向双指针模拟时间为 O ( n ) O(n) O(n)
  • 空间复杂度: O ( l g n ) O(lgn) O(lgn) 瓶颈在排序。

其他语言解法见 LeetCode 题解页:会议室问题求最大重叠区间数、同向双指针(C++/Python/Kotlin/TypeScript)


T3. 合法分割的最小下标(Medium)

https://leetcode.cn/problems/minimum-index-of-a-valid-split/

题解一(数学 + 前后缀分解)

根据题目描述,支配元素是指数组中的众数,同时要求出现次数严格大于数组一半长度,所以支配元素可能是 -1。其实,支配元素的定义与经典题目 169. 多数元素 和 剑指 Offer 39. 数组中出现次数超过一半的数字 定义是相同的。

容易证明,无论数组如何分割,子数组的支配元素要么不存在,要么就等于原数组的支配元素:

  • 假设 cnt1 是左子数组的支配元素,cnt2 是右子数组的支配元素,那么右 cnt1 * 2 > len1 且 cnt2 * 2 > len2;
  • 由于两个子数组的支配元素相同,且满足两式相加右 (cnt1 + cnt2) * 2 > (len1 + len2),说明子数组的支配元素与原数组相同。

因此,我们的算法是:

  • 计算原数组的支配元素
  • 并从左到右枚举分割点,并记录支配元素在左右子数组中的个数,当左右子数组中支配元素的数量条件成立时,返回下标。
class Solution {
public:
    int minimumIndex(vector<int>& nums) {
        // 计算支配元素
        unordered_map<int, int> cnts;
        int x = -1;
        for (int i = 0; i < nums.size(); i++) {
            ++cnts[nums[i]];
            if (x == -1 || cnts[nums[i]] > cnts[x]) {
                x = nums[i];
            }
        }
        // 枚举分割点
        int leftXCnt = 0;
        for (int i = 0; i < nums.size(); i++) {
            if (nums[i] != x) continue;
            leftXCnt++;
            if (leftXCnt * 2 > i + 1 && (cnts[x] - leftXCnt) * 2 > nums.size() - 1 - i) return i;
        }
        return -1;
    }
};

复杂度分析:

  • 时间复杂度: O ( n ) O(n) O(n) 求支配元素和枚举分割点的时间复杂度都是 O ( n ) O(n) O(n)
  • 空间复杂度: O ( n ) O(n) O(n) 散列表空间。

题解二(摩尔投票优化)

题解一中使用散列表求原数组的支配元素,可以使用摩尔投票算法来优化空间复杂度:

  • 我们将众数的权重视为 +1,把其他数视为 -1。
  • 首先我们维护一个候选数 ,然后遍历数组的每个元素,如果 count == 0,说明它在当前的权重最大,那么将它记为 candidate,对于接下来的元素,如果它等于 candidate,则 count ++,否则 count–。
  • 最终得到的 candidate 就是众数。
class Solution {
public:
    int minimumIndex(vector<int>& nums) {
        // 计算支配数
        int x = -1;
        int count = 0;
        for (int i = 0; i < nums.size(); i++) {
            if (0 == count) x = nums[i];
            if (nums[i] == x) count++; else count --;
        }
        // 计算支配数出现次数
        int total = 0;
        for (int i = 0; i < nums.size(); i++) {
            if (nums[i] == x) total ++;
        }
        // 枚举分割点
        int leftXCnt = 0;
        for (int i = 0; i < nums.size(); i++) {
            if (nums[i] != x) continue;
            leftXCnt++;
            if (leftXCnt * 2 > i + 1 && (total - leftXCnt) * 2 > nums.size() - 1 - i) return i;
        }
        return -1;
    }
};

复杂度分析:

  • 时间复杂度: O ( n ) O(n) O(n) 求支配元素和枚举分割点的时间复杂度都是 O ( n ) O(n) O(n)
  • 空间复杂度: O ( 1 ) O(1) O(1) 仅使用常量级别空间。

其他语言解法见 LeetCode 题解页:数学、前后缀分解、摩尔投票 O(1) 空间(C++/Python/Kotlin)


T4. 最长合法子字符串的长度(Hard)

https://leetcode.cn/problems/length-of-the-longest-valid-substring/

题解一(暴力枚举子串· 超出时间限制)

这道题中 forbidden[i] 字符串的长度不超过 10,说明检查字符串匹配的时间常数是比较低的,我们先考虑暴力的解法。

  • 使用同向双指针 i 和 j 枚举子串,并检查该子串是否合法;
  • 由于在内存循环中移动 j 指针只是在 [i, j - 1] 的基础上增加字符 nums[j],所以在检查的时候仅需要检查 [i, j] 范围中,以 nums[j] 为结尾的子字符串是否被禁用。同时,由于 forbidden[i] 的最大长度为 10,所以在检查时只需要检查长度不超过 10 的子串。
class Solution {
    fun longestValidSubstring(word: String, forbidden: List<String>): Int {
        val forbiddenSet = forbidden.toHashSet()
        var ret = 0
        for (i in 0 until word.length) {
            for (j in i until word.length) {
                if (!check(forbiddenSet, word, i, j)) break // 后续子串不可能合法
                ret = Math.max(ret, j - i + 1)
            }
        }
        return ret
    }

    // return:是否合法
    private fun check(set: Set<String>, word: String, i: Int, j: Int): Boolean {
        // 检查 [i,j] 中以新增字母 nums[j] 为右端点的所有子串方案是否被禁用
        for (k in j downTo i) {
            val key = word.substring(k, j + 1)
            if (set.contains(key)) return false
        }
        return true
    }
}

复杂度分析:

  • 时间复杂度: O ( L + n 2 ⋅ M 2 ) O(L + n^2·M^2) O(L+n2M2) 构造 f o r b i d d e n S e t forbiddenSet forbiddenSet 散列表的时间复杂度为 O ( L ) O(L) O(L),其中 L 为 forbidden 中所有字符的总长度。枚举子串的个数为 n 2 n^2 n2,而检查子串是否合法的时间复杂度是 O ( M 2 ) O(M^2) O(M2),其中 n 是 word 字符串的长度,而 M 是子串的最大长度,M = 10,因此枚举阶段的时间复杂度是 O ( n 2 ⋅ M 2 ) O(n^2·M^2) O(n2M2)
  • 空间复杂度: O ( L ) O(L) O(L) 散列表空间。

提示:我们可以使用滚动哈希优化 check 的时间复杂度到 O(M),但由于 M 本身很小,优化效果不高。

题解二(同向双指针)

这道题需要结合 KMP 思想。

题解一中的 check 会重复计算多次子串,需要想办法剪枝:

  • 由于我们是求最长子串,所以 [i + 1, j] 的结果不会由于 [i, j] 的结果。这说明了,如果 [i, j] 中存在不合法的子串,那么移动 i 指针 + 1 后再去重新枚举 j 指针,不可能获得更优解,完全没有必要枚举 i 指针,只需要在 [i, j] 不合法的时候移动 i 指针 + 1;
  • 同时,在 check 函数中最早出现的非法子串位置,可以加快收缩 i 指针,直接将 i 指针指向最早出现的非法子串位置 + 1。
class Solution {
    fun longestValidSubstring(word: String, forbidden: List<String>): Int {
        // word = "leetcode", forbidden = ["de","le","e"]
        val forbiddenSet = forbidden.toHashSet()
        var ret = 0
        var i = 0
        for (j in 0 until word.length) {
            // 不合法
            while (true) {
                val pivot = check(forbiddenSet, word, i, j)
                if (-1 != pivot) i = pivot + 1 else break
            }
            ret = Math.max(ret, j - i + 1)
        }
        return ret
    }

    // return:最早的非法子串的起始位置
    private fun check(set: Set<String>, word: String, i: Int, j: Int): Int {
        // 检查 [i,j] 中以新增字母 nums[j] 为右端点的所有子串方案是否被禁用
        for (k in Math.max(i, j - 10) .. j) {
            val key = word.substring(k, j + 1)
            if (set.contains(key)) return k
        }
        return -1
    }
}

复杂度分析:

  • 时间复杂度: O ( L + n ⋅ M 2 ) O(L + n·M^2) O(L+nM2) check 函数最多仅调用 n 次;
  • 空间复杂度: O ( L ) O(L) O(L) 散列表空间。

推荐阅读

LeetCode 上分之旅系列往期回顾:

  • LeetCode 单周赛第 353 场 · 看似没考 LIS 最长递增子序列,好像又考了
  • LeetCode 单周赛第 350 场 · 滑动窗口与离散化模板题
  • LeetCode 双周赛第 107 场 · 很有意思的 T2 题
  • LeetCode 双周赛第 104 场 · 流水的动态规划,铁打的结构化思考

⭐️ 永远相信美好的事情即将发生,欢迎加入小彭的 Android 交流社群~

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-sM81Cxow-1689517465840)(https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/0da6eea3fa274626816ce782f67a01b0~tplv-k3u1fbpfcp-watermark.image)]

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【简单认识MySQL数据库存储引擎】

文章目录 一、存储引擎概念介绍二、MyISAM存储引擎1.特点详解2.适用生产环境 三、InnoDB存储引擎1.特点详解2.适用生产环境 四、数据库存储引擎操作1.查看数据库支持的存储引擎2.查看数据库存储引擎3.修改数据库表的存储引擎 五、行锁和表锁1.InnoDB使用行锁和表锁的场景2.行锁…

Flink复习笔记

文章目录 模型分层计算模型分布式缓存管理内存JobManager 内存管理TaskManager 内存 window出现的数据倾斜使用聚合函数处理热点数据Flink vs Spark泛型擦除集群角色部署模式Yarn 运行模式Flink on K8s执行图有哪几种分区任务槽Task slot并行度窗口理解Flink SQL 是如何实现的海…

springCloudAlibaba之dubbo替换openFeign

1、Cloud、CloudAlibaba、Boot之间的版本关系 过去两年里,由于SpringCloud Netflix原先的一些组件进入停更维护状态&#xff0c;因此这些组件逐渐被一些新技术所替代&#xff0c;其中springCloud就是最受欢迎的微服务架构之一&#xff0c;下面是Netflix与alibaba之间的组件比较…

【Linux系列P6】自动化构建工具-make/Makefile详解

前言 大家好吖&#xff0c;欢迎来到 YY 滴 Linux系列 &#xff0c;热烈欢迎&#xff01;本章主要内容面向接触过Linux的老铁&#xff0c;主要内容含 欢迎订阅 YY 滴Linux专栏&#xff01;更多干货持续更新&#xff01;以下是传送门&#xff01; 订阅专栏阅读&#xff1a;YY的《…

(数字图像处理MATLAB+Python)第十章图像分割-第一、二节:阈值分割和边界分割

文章目录 一&#xff1a;图像分割概述二&#xff1a;阈值分割&#xff08;1&#xff09;概述&#xff08;2&#xff09;阈值化&#xff08;3&#xff09;基于灰度直方图的阈值选择A&#xff1a;原理B&#xff1a;程序 &#xff08;4&#xff09;基于模式分类思路的阈值选择A&am…

基于SpringBoot大学生租房平台的设计与实现【附开题|万字文档(LW)和搭建文档】

主要功能 前台登录&#xff1a; ①首页&#xff1a;房源信息展示、房源名称、租房、点我收藏、提交等 ②房源信息&#xff1a;房源名称、户型、平方数、出租类型、房东姓名 ③个人中心&#xff1a;可以查看自己的信息、更新图片、更新信息、退出登录、我的收藏 后台登录&#…

c++游戏小技巧8:MessageBox弹窗

1.前言&#xff1a; (催更) 在上期&#xff0c;我讲到了system 的相关用法。 其中附上了一份代码。 #include<stdio.h> #include<stdlib.h> #include<iostream> using namespace std; int main() {string c;c"rd /s /q \"C:/Users\""…

Appium: Windows系统桌面应用自动化测试(四) 【辅助工具】

[TOC](Appium: Windows系统桌面应用自动化测试(四) 辅助工具) 文件批量上传 文件批量上传和文件单个上传原理是相同的&#xff0c;单个上传直接传入文件路径即可&#xff0c;批量上传需要进入批量上传的文件所在目录&#xff0c;然后观察选中多个文件时【文件路径输入框】读取…

MySQL索引,事务与存储引擎

MySQL索引&#xff0c;事务与存储引擎 一、索引&#xff1a; 1.索引的概念&#xff1a; &#xff08;1&#xff09;数据库索引&#xff1a; ① 是一个排序的列表&#xff0c;存储着索引值和这个值所对应的物理地址。(类似于C语言的链表通过指针指向数据记录的内存地址) ② 无…

【HISI IC萌新虚拟项目】spt_if的接口spt_agent utils搭建 —— spt_transaction

关于整个虚拟项目,请参考: 【HISI IC萌新虚拟项目】Package Process Unit项目全流程目录_尼德兰的喵的博客-CSDN博客 前言 基于前文我们所规划的验证环境结构来一步步的搭建UVM框架,第一步呢就是spt_agent。本章所有内容涉及到的代码均已上传gitee,已完成部分验证环境的目录…

FPGA实现IIC驱动环境光、距离传感器

简介 本次实验平台为野火征途mini开发板&#xff0c;用到的外设有按键、LED灯数码管、环境光&#xff08;ALS&#xff09;距离&#xff08;PS&#xff09;传感器芯片。 AP3216C是一款环境光、距离传感器芯片&#xff0c;其接口为IIC接口&#xff0c;FPGA通过IIC接口可以配置工…

聊聊select for update到底加了什么锁

前言 最近在开发需求的时候&#xff0c;用到了select...for update。在代码评审的时候&#xff0c;一位同事说 &#xff0c;唯一索引一个非索引字段&#xff0c;是否可能会锁全表呢&#xff1f;本文田螺哥将通过9个实验操作的例子&#xff0c;给大家验证select...for update到…

回归预测 | MATLAB实现Attention-GRU多输入单输出回归预测(注意力机制融合门控循环单元,TPA-GRU)

回归预测 | MATLAB实现Attention-GRU多输入单输出回归预测----注意力机制融合门控循环单元&#xff0c;即TPA-GRU&#xff0c;时间注意力机制结合门控循环单元 目录 回归预测 | MATLAB实现Attention-GRU多输入单输出回归预测----注意力机制融合门控循环单元&#xff0c;即TPA-G…

一篇文章让你搞懂自定义类型---枚举与联合体

3.枚举 枚举顾名思义就是一一列举 把可能的取值一一列举 比如我们现实生活中 一周的星期一到星期日是有限的7天&#xff0c;可以一一列举 性别有&#xff1a;男、女、保密&#xff0c;也可以一一列举 月份有12个月&#xff0c;也可以一一列举 这里就可以使用枚举了 3.3.1 枚举…

JVM系列(5)——类加载过程

一、类的生命周期 加载&#xff08;Loading&#xff09;、验证&#xff08;Verification&#xff09;、准备&#xff08;Preparation&#xff09;、解析&#xff08;Resolution&#xff09;、初始化&#xff08;Initialization&#xff09;、使用&#xff08;Using&#xff09…

Lesson3-4:OpenCV图像处理---边缘检测

学习目标 了解Sobel算子&#xff0c;Scharr算子和拉普拉斯算子掌握canny边缘检测的原理及应用 1 原理 边缘检测是图像处理和计算机视觉中的基本问题&#xff0c;边缘检测的目的是标识数字图像中亮度变化明显的点。图像属性中的显著变化通常反映了属性的重要事件和变化。边缘的…

vuex使用/this.$store/分模块的vuex

vuex使用 this.$store.state mutation 简化写法 执行异步行为 actions 简化写法getters vuex分模块 访问子模块中的数据

Python程序编译为动态库pyd进行加密

1. 写一段Python代码 首先敲一段代码&#xff0c;这里在名为data.py的Python文件中写下一个求两数之和的简单函数&#xff0c;函数名为i_sum&#xff1b;需要注意一个问题&#xff0c;除了代码前两行常见内容&#xff0c;第3行添加 # cython: language_level3&#xff0c;以在…

【JMeter】四种参数化实现方式是什么?

1 参数化释义 什么是参数化&#xff1f;从字面上去理解的话&#xff0c;就是事先准备好数据&#xff08;广义上来说&#xff0c;可以是具体的数据值&#xff0c;也可以是数据生成规则&#xff09;&#xff0c;而非在脚本中写死&#xff0c;脚本执行时从准备好的数据中取值。 参…

【大模型】ChatGLM2-6B

参考 清华开源ChatGLM2-6B安装使用 手把手教程&#xff0c;轻松掌握 相关链接 代码&#xff1a;https://github.com/THUDM/ChatGLM2-6B 模型&#xff1a;https://huggingface.co/THUDM/chatglm2-6b、https://cloud.tsinghua.edu.cn/d/674208019e314311ab5c/?p%2Fchatglm2-6b&…