【码道初阶】一道经典简单题:多数元素(LeetCode 169)|Boyer-Moore 投票算法详解

多数元素问题是算法面试中最经典的题之一:

给定数组 nums,其中一个元素的出现次数 超过 n/2 ,找出它。

要求:

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

示例:

复制代码
输入: [2,2,1,1,1,2,2]
输出: 2

题目保证多数元素一定存在。


一、核心思想:成对抵消

把数组看成选票,每个数字代表一个候选人。当两个不同的元素相遇时,可以视为一张支持票和一张反对票抵消了。

这就是把直接找最多的元素这个问题,转化成了数组间元素互相竞争计票的问题,思想上能转过这个弯来想到这个点,这便是一道Easy题。

由于多数元素(majority element)的票数大于其它所有元素票数的总和,它无法被完全抵消,因此最终剩下的候选人一定是多数元素。

这个思想催生了著名的 ------ Boyer--Moore Majority Vote Algorithm


二、实现步骤

首先我们要创建两个变量:

  • candidate:当前候选人
  • count:候选人当前的"净票数"

遍历数组的每个元素 num

  1. 如果 count == 0

    说明之前的票已经两两抵消完,可以重选候选人:

    复制代码
    candidate = num
    count = 1
  2. 如果 num == candidate

    复制代码
    count++
  3. 否则:

    复制代码
    count--

最终的 candidate 就是多数元素。


三、正确 Java 实现(O(n) 时间 + O(1) 空间)

java 复制代码
class Solution {
    public int majorityElement(int[] nums) {
        int candidate = 0;
        int count = 0;

        for (int num : nums) {
            if (count == 0) {
                candidate = num;
                count = 1;
            } else if (num == candidate) {
                count++;
            } else {
                count--;
            }
        }

        return candidate;
    }
}

这段代码能通过所有测试用例,是多数投票算法的标准写法。


四、特别提醒(⚠ 初学者最常犯的致命错误)

在实现多数投票算法时,许多同学会错误地写出下面这种版本:

java 复制代码
if (count == 0) {
    candidate = num;
    count = 0;      // ❌ 致命错误!
}

这段代码的问题是:

当 count == 0 且选中新候选人时,却不给它加上当前这一票。

结果是:

  • count 一直保持为 0
  • 根本没有进行投票累积
  • 每次都只是不断更换候选人
  • 于是算法最终返回的是数组的最后一个元素
  • 在测试用例 [3,3,4] 中错误返回 4

正确做法必须是:

java 复制代码
count = 1;   // 当前元素至少贡献一票

这是 Boyer-Moore 算法最关键的地方。

如果这一点写错,整个算法将完全失效。


五、为什么算法可靠?

多数元素出现次数 > n/2,因此:

  • 它的票数比所有其他元素加起来还多
  • 在"抵消"过程中,成对消掉的都是多数元素与非多数元素的票
  • 多数元素永远不可能被完全抵消
  • 并且当 count 归零时,多数元素在数组后面部分必然还会再次出现,重新成为 candidate

因此,算法仅需一次遍历即可确定结果。


六、总结

Boyer-Moore 投票算法是一种极其优雅的线性时间算法,特点:

  • 时间复杂度:O(n)(单次遍历)
  • 空间复杂度:O(1)(仅用两个变量)
  • 适用场景:明确存在多数元素

并且务必牢记:

⚠ 实现时 count = 0 分支必须写成 count = 1,否则算法彻底失效。

掌握了这个技巧,许多相关问题(比如"找出现超过 n/3 的元素"、"找出现 k 次的元素")都能顺藤摸瓜继续拓展。


另一种思路(更简洁):

因为存在一定出现>n/2次的元素,那么对数组进行排序后,在中间的元素就是要找的多数元素。

java 复制代码
import java.util.Arrays;
class Solution {
    public int majorityElement(int[] nums) {
        Arrays.sort(nums);
        return nums[nums.length/2];
    }
}
相关推荐
wadesir1 小时前
C语言模块化设计入门指南(从零开始构建清晰可维护的C程序)
c语言·开发语言·算法
t198751281 小时前
MATLAB水声信道仿真程序
开发语言·算法·matlab
CoderYanger2 小时前
动态规划算法-简单多状态dp问题:15.买卖股票的最佳时机含冷冻期
开发语言·算法·leetcode·动态规划·1024程序员节
Xの哲學2 小时前
Linux RTC深度剖析:从硬件原理到驱动实践
linux·服务器·算法·架构·边缘计算
狐573 小时前
2025-12-04-牛客刷题笔记-25_12-4-质数统计
笔记·算法
小O的算法实验室3 小时前
2024年IEEE IOTJ SCI2区TOP,基于混合算法的水下物联网多AUV未知环境全覆盖搜索方法,深度解析+性能实测
算法·论文复现·智能算法·智能算法改进
洲星河ZXH3 小时前
Java,比较器
java·开发语言·算法
尋有緣3 小时前
力扣1069-产品销售分析II
leetcode·oracle·数据库开发
CoderYanger3 小时前
递归、搜索与回溯-FloodFill:33.太平洋大西洋水流问题
java·算法·leetcode·1024程序员节