LeetCode 面试题 17.10. Find Majority Element LCCI【摩尔投票法】简单

本文主要是介绍LeetCode 面试题 17.10. Find Majority Element LCCI【摩尔投票法】简单,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文属于「征服LeetCode」系列文章之一,这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁,本系列将至少持续到刷完所有无锁题之日为止;由于LeetCode还在不断地创建新题,本系列的终止日期可能是永远。在这一系列刷题文章中,我不仅会讲解多种解题思路及其优化,还会用多种编程语言实现题解,涉及到通用解法时更将归纳总结出相应的算法模板。

为了方便在PC上运行调试、分享代码文件,我还建立了相关的仓库:https://github.com/memcpy0/LeetCode-Conquest。在这一仓库中,你不仅可以看到LeetCode原题链接、题解代码、题解文章链接、同类题目归纳、通用解法总结等,还可以看到原题出现频率和相关企业等重要信息。如果有其他优选题解,还可以一同分享给他人。

由于本系列文章的内容随时可能发生更新变动,欢迎关注和收藏征服LeetCode系列文章目录一文以作备忘。

数组中占比超过一半的元素称之为主要元素。给你一个 整数 数组,找出其中的主要元素。若没有,返回 -1 。请设计时间复杂度为 O(N) 、空间复杂度为 O(1) 的解决方案。

示例 1:

输入:[1,2,5,9,5,9,5,5,5]
输出:5

示例 2:

输入:[3,2]
输出:-1

示例 3:

输入:[2,2,1,1,1,2,2]
输出:2

题目集合:

  • 169. 多数元素
  • 面试题 17.10. Find Majority Element LCCI
  • 229. 多数元素 II
    1. Check If a Number Is Majority Element in a Sorted Array
  • 1157. 子数组中占绝大多数的元素:困难

解法 Boyer-Moore 投票算法

由于题目要求时间复杂度 O ( n ) O(n) O(n) 和空间复杂度 O ( 1 ) O(1) O(1) ,因此符合要求的解法只有 Boyer-Moore 投票算法。这一投票算法在求出现次数大于 ⌊ n / 2 ⌋ \lfloor n / 2 \rfloor n/2 的元素 x x x 时很好理解:如果我们把 x x x 记为 + 1 +1 +1 ,把其他数记为 − 1 -1 1 ,将它们全部加起来,显然和大于 0 0 0 ,从结果本身我们可以看出 x x x 比其他数多。

我们首先给出 Boyer-Moore 算法的详细步骤:

  1. 维护一个候选主要元素 c a n d i d a t e candidate candidate 和候选主要元素的出现次数 c o u n t count count初始时 c a n d i d a t e candidate candidate 为任意值 c o u n t = 0 count=0 count=0
  2. 遍历数组 nums \textit{nums} nums 中的所有元素,遍历到元素 x x x,在判断 x x x 之前,如果 c o u n t = 0 count=0 count=0 ,我们先将 x x x 的值赋给 c a n d i d a t e candidate candidate ,否则不更新 c a n d i d a t e candidate candidate 的值。随后我们判断 x x x
    1. 如果 x = c a n d i d a t e x=candidate x=candidate ,则将 c o u n t count count 1 1 1
    2. 如果 x ≠ c a n d i d a t e x\ne candidate x=candidate ,则将 c o u n t count count 1 1 1
  3. 遍历结束之后,如果数组 n u m s nums nums 中存在主要元素,则 c a n d i d a t e candidate candidate 即为主要元素,否则 c a n d i d a t e candidate candidate 可能为数组中的任意一个元素。
  4. 由于不一定存在主要元素,因此需要第二次遍历数组,验证 c a n d i d a t e candidate candidate 是否为主要元素。第二次遍历时,统计 c a n d i d a t e candidate candidate 在数组中的出现次数,如果出现次数大于数组长度的一半,则 c a n d i d a t e candidate candidate 是主要元素,返回 c a n d i d a t e candidate candidate ,否则数组中不存在主要元素,返回 − 1 -1 1

举一个具体的例子,例如下面的这个数组:

[7, 7, 5, 7, 5, 1 | 5, 7 | 5, 5, 7, 7 | 7, 7, 7, 7]

在遍历到数组中的第一个元素以及每个在 | 之后的元素时, c a n d i d a t e candidate candidate 都会因为 c o u n t count count 的值变为 0 0 0 而发生改变。最后一次 c a n d i d a t e candidate candidate 的值从 5 5 5 变为 7 7 7 ,也就是这个数组中的主要元素。

Boyer-Moore 算法的正确性较难证明,这里给出一种较为详细的用例子辅助证明的思路,供参考:
1.首先根据算法步骤中对 c o u n t count count 的定义,可以发现:在对整个数组进行遍历的过程中, c o u n t count count 的值一定非负。这是因为==如果 c o u n t count count 的值为 0 0 0 ,那么在这一轮遍历的开始时刻,我们会将 x x x 的值赋予 c a n d i d a t e candidate candidate 并在接下来的一步中将 c o u n t count count 的值增加 1 1 1 ==。因此 c o u n t count count 的值在遍历过程中一直保持非负

2.那么 c o u n t count count 本身除了计数器之外,还有什么更深层次的意义呢?我们还是以数组:

[7, 7, 5, 7, 5, 1 | 5, 7 | 5, 5, 7, 7 | 7, 7, 7, 7]

作为例子,首先写下它在每一步遍历时 c a n d i d a t e candidate candidate c o u n t count count 的值:

nums:      [7, 7, 5, 7, 5, 1 | 5, 7 | 5, 5, 7, 7 | 7, 7, 7, 7]
candidate:  7  7  7  7  7  7   5  5   5  5  5  5   7  7  7  7
count:      1  2  1  2  1  0   1  0   1  2  1  0   1  2  3  4

我们再定义一个变量 v a l u e value value ,它和真正的主要元素 m a j maj maj 绑定。在每一步遍历时,如果当前的数 x x x m a j maj maj 相等,那么 v a l u e value value 的值加 1 1 1 ,否则减 1 1 1 v a l u e value value 的实际意义即为:到当前的这一步遍历为止,主要元素出现的次数比非主要元素多出了多少次。我们将 v a l u e value value 的值也写在下方:

nums:      [7, 7, 5, 7, 5, 1 | 5, 7 | 5, 5, 7, 7 | 7, 7, 7, 7]
value:      1  2  1  2  1  0  -1  0  -1 -2 -1  0   1  2  3  4

有没有发现什么?我们将 c o u n t count count v a l u e value value 放在一起:

nums:      [7, 7, 5, 7, 5, 1 | 5, 7 | 5, 5, 7, 7 | 7, 7, 7, 7]
count:      1  2  1  2  1  0   1  0   1  2  1  0   1  2  3  4
value:      1  2  1  2  1  0  -1  0  -1 -2 -1  0   1  2  3  4

发现在每一步遍历中, c o u n t count count v a l u e value value 要么相等,要么互为相反数!并且在候选主要元素 c a n d i d a t e candidate candidate 就是 m a j maj maj 时,它们相等, c a n d i d a t e candidate candidate 是其它的数时,它们互为相反数!

为什么会有这么奇妙的性质呢?这并不难证明:我们将候选主要元素 c a n d i d a t e candidate candidate 保持不变的连续的遍历称为「一段」。在同一段中, c o u n t count count 的值是根据 c a n d i d a t e = = x candidate == x candidate==x 的判断进行加减的。那么如果 c a n d i d a t e candidate candidate 恰好为 m a j maj maj ,那么在这一段中, c o u n t count count v a l u e value value 的变化是同步的;如果 c a n d i d a t e candidate candidate 不为 m a j maj maj ,那么在这一段中 c o u n t count count v a l u e value value 的变化是相反的。因此就有了这样一个奇妙的性质。

这样以来,由于:

  • 我们证明了 c o u n t count count 的值一直为非负,在最后一步遍历结束后也是如此;
  • 由于 v a l u e value value 的值与真正的主要元素 m a j maj maj 绑定,并且它表示「主要元素出现的次数比非主要元素多出了多少次」,那么在最后一步遍历结束后, v a l u e value value 的值为正数
  • 在最后一步遍历结束后, c o u n t count count 非负, v a l u e value value 为正数,所以它们不可能互为相反数,只可能相等,即 c o u n t = = v a l u e count == value count==value 。因此在最后「一段」中, c o u n t count count v a l u e value value 的变化是同步的,也就是说, c a n d i d a t e candidate candidate 中存储的候选主要元素就是真正的主要元素 m a j maj maj
class Solution {
public:int majorityElement(vector<int>& nums) {int candidate = 0, count = 0;for (int num : nums) {if (count == 0) candidate = num;if (candidate == num) ++count;else --count;}count = 0;for (int num : nums) if (num == candidate) ++count;return count * 2 > nums.size() ? candidate : -1;}
};

复杂度分析:

  • 时间复杂度: O ( n ) O(n) O(n) ,其中 n n n 是数组 n u m s nums nums 的长度。
  • 空间复杂度: O ( 1 ) O(1) O(1) 。只需要常数的额外空间。

这篇关于LeetCode 面试题 17.10. Find Majority Element LCCI【摩尔投票法】简单的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/163930

相关文章

哈希leetcode-1

目录 1前言 2.例题  2.1两数之和 2.2判断是否互为字符重排 2.3存在重复元素1 2.4存在重复元素2 2.5字母异位词分组 1前言 哈希表主要是适合于快速查找某个元素(O(1)) 当我们要频繁的查找某个元素,第一哈希表O(1),第二,二分O(log n) 一般可以分为语言自带的容器哈希和用数组模拟的简易哈希。 最简单的比如数组模拟字符存储,只要开26个c

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

hdu2289(简单二分)

虽说是简单二分,但是我还是wa死了  题意:已知圆台的体积,求高度 首先要知道圆台体积怎么求:设上下底的半径分别为r1,r2,高为h,V = PI*(r1*r1+r1*r2+r2*r2)*h/3 然后以h进行二分 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#includ

usaco 1.3 Prime Cryptarithm(简单哈希表暴搜剪枝)

思路: 1. 用一个 hash[ ] 数组存放输入的数字,令 hash[ tmp ]=1 。 2. 一个自定义函数 check( ) ,检查各位是否为输入的数字。 3. 暴搜。第一行数从 100到999,第二行数从 10到99。 4. 剪枝。 代码: /*ID: who jayLANG: C++TASK: crypt1*/#include<stdio.h>bool h

uva 10387 Billiard(简单几何)

题意是一个球从矩形的中点出发,告诉你小球与矩形两条边的碰撞次数与小球回到原点的时间,求小球出发时的角度和小球的速度。 简单的几何问题,小球每与竖边碰撞一次,向右扩展一个相同的矩形;每与横边碰撞一次,向上扩展一个相同的矩形。 可以发现,扩展矩形的路径和在当前矩形中的每一段路径相同,当小球回到出发点时,一条直线的路径刚好经过最后一个扩展矩形的中心点。 最后扩展的路径和横边竖边恰好组成一个直

poj 1113 凸包+简单几何计算

题意: 给N个平面上的点,现在要在离点外L米处建城墙,使得城墙把所有点都包含进去且城墙的长度最短。 解析: 韬哥出的某次训练赛上A出的第一道计算几何,算是大水题吧。 用convexhull算法把凸包求出来,然后加加减减就A了。 计算见下图: 好久没玩画图了啊好开心。 代码: #include <iostream>#include <cstdio>#inclu

uva 10130 简单背包

题意: 背包和 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstring>#include <cmath>#include <stack>#include <vector>#include <queue>#include <map>

荣耀嵌入式面试题及参考答案

在项目中是否有使用过实时操作系统? 在我参与的项目中,有使用过实时操作系统。实时操作系统(RTOS)在对时间要求严格的应用场景中具有重要作用。我曾参与的一个工业自动化控制项目就采用了实时操作系统。在这个项目中,需要对多个传感器的数据进行实时采集和处理,并根据采集到的数据及时控制执行机构的动作。实时操作系统能够提供确定性的响应时间,确保关键任务在规定的时间内完成。 使用实时操作系统的

leetcode-24Swap Nodes in Pairs

带头结点。 /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode(int x) { val = x; }* }*/public class Solution {public ListNode swapPairs(L

leetcode-23Merge k Sorted Lists

带头结点。 /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode(int x) { val = x; }* }*/public class Solution {public ListNode mergeKLists