摩尔投票算法原理实现一文剖析

news2025/6/4 12:59:06

摩尔投票算法原理&实现一文剖析

    • 一、算法原理
      • 1.1 基本思想
      • 1.2 数学原理
    • 二、算法实现
      • 2.1 Python实现
      • 2.2 Java实现
      • 2.3 C++实现
    • 三、复杂度分析
    • 四、应用场景
      • 4.1 多数元素问题
      • 4.2 扩展应用:寻找出现次数超过n/3的元素
    • 五、算法优势与注意事项
      • 5.1 优势
      • 5.2 注意事项
    • 总结

摩尔投票法(Boyer-Moore Voting Algorithm)是一种高效的算法,主要用于在数组中寻找出现次数超过一半的元素,即多数元素。该算法由Robert S. Boyer和J Strother Moore在1981年提出,其时间复杂度为O(n),空间复杂度为O(1),是处理这类问题的最优解法。本文我将为你详细介绍摩尔投票法的原理、实现及应用场景。

一、算法原理

1.1 基本思想

摩尔投票法的核心思想是基于这样一个观察:如果一个元素在数组中出现次数超过一半,那么该元素的出现次数一定比其他所有元素的出现次数总和还要多。

算法通过维护两个变量来实现:

  • 候选元素(candidate):记录当前可能的多数元素
  • 计数(count):记录候选元素的相对出现次数

算法的执行过程可以分为两个阶段:

  1. 投票阶段:遍历数组,对每个元素进行投票。如果当前元素与候选元素相同,则计数加1;否则计数减1。当计数减为0时,更换候选元素为当前元素,并将计数重置为1。
  2. 验证阶段:遍历结束后,得到的候选元素需要再次验证是否真的出现次数超过一半(在某些题目中,可能需要这一步骤,但在明确存在多数元素的情况下可以省略)。

1.2 数学原理

假设数组长度为n,多数元素出现次数为m(m > n/2)。在投票过程中,每当遇到一个非候选元素时,计数减1,但由于多数元素的出现次数超过一半,即使每次遇到非候选元素都抵消一次,最终候选元素的计数仍会大于0。因此,最终得到的候选元素必然是多数元素。
摩尔投票法

二、算法实现

2.1 Python实现

def majorityElement(nums):
    """
    摩尔投票法实现,寻找数组中出现次数超过一半的元素
    """
    # 初始化候选元素和计数
    candidate = None
    count = 0
    
    # 投票阶段
    for num in nums:
        if count == 0:
            # 当计数为0时,更换候选元素为当前元素
            candidate = num
            count = 1
        elif num == candidate:
            # 当前元素与候选元素相同,计数加1
            count += 1
        else:
            # 当前元素与候选元素不同,计数减1
            count -= 1
    
    # 验证阶段(在明确存在多数元素的情况下可以省略)
    # 这里简单验证候选元素的出现次数是否超过一半
    count = 0
    for num in nums:
        if num == candidate:
            count += 1
    if count > len(nums) // 2:
        return candidate
    else:
        return None  # 实际上题目保证存在多数元素,不会执行到这一步

# 示例用法
nums = [2, 2, 1, 1, 1, 2, 2]
print("多数元素是:", majorityElement(nums))  # 输出: 2

2.2 Java实现

public class BoyerMooreVoting {
    public static 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--;
            }
        }
        
        // 验证阶段(在明确存在多数元素的情况下可以省略)
        count = 0;
        for (int num : nums) {
            if (num == candidate) {
                count++;
            }
        }
        if (count > nums.length / 2) {
            return candidate;
        } else {
            return -1;  // 实际上题目保证存在多数元素,不会执行到这一步
        }
    }
    
    public static void main(String[] args) {
        int[] nums = {2, 2, 1, 1, 1, 2, 2};
        System.out.println("多数元素是: " + majorityElement(nums));  // 输出: 2
    }
}

2.3 C++实现

#include <iostream>
#include <vector>
using namespace std;

int majorityElement(vector<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--;
        }
    }
    
    // 验证阶段(在明确存在多数元素的情况下可以省略)
    count = 0;
    for (int num : nums) {
        if (num == candidate) {
            count++;
        }
    }
    if (count > nums.size() / 2) {
        return candidate;
    } else {
        return -1;  // 实际上题目保证存在多数元素,不会执行到这一步
    }
}

int main() {
    vector<int> nums = {2, 2, 1, 1, 1, 2, 2};
    cout << "多数元素是: " << majorityElement(nums) << endl;  // 输出: 2
    return 0;
}

三、复杂度分析

  • 时间复杂度:O(n),算法只需遍历数组两次(投票阶段一次,验证阶段一次),每次遍历的时间复杂度均为O(n)。
  • 空间复杂度:O(1),只需要常数级的额外空间来存储候选元素和计数。

四、应用场景

4.1 多数元素问题

摩尔投票法最典型的应用是解决LeetCode上的"多数元素"问题(题目编号169):给定一个大小为n的数组,找到其中的多数元素。多数元素是指在数组中出现次数大于⌊n/2⌋的元素。

4.2 扩展应用:寻找出现次数超过n/3的元素

摩尔投票法可以扩展用于寻找数组中出现次数超过n/3的元素(LeetCode题目编号229)。此时需要维护两个候选元素和对应的计数,基本思想类似,但在投票阶段需要更复杂的逻辑处理。

def majorityElement(nums):
    """
    寻找数组中出现次数超过n/3的元素
    """
    if not nums:
        return []
    
    # 初始化两个候选元素和计数
    candidate1, candidate2 = None, None
    count1, count2 = 0, 0
    
    # 投票阶段
    for num in nums:
        if num == candidate1:
            count1 += 1
        elif num == candidate2:
            count2 += 1
        elif count1 == 0:
            candidate1 = num
            count1 = 1
        elif count2 == 0:
            candidate2 = num
            count2 = 1
        else:
            count1 -= 1
            count2 -= 1
    
    # 验证阶段
    result = []
    threshold = len(nums) // 3
    for candidate in [candidate1, candidate2]:
        if nums.count(candidate) > threshold:
            result.append(candidate)
    
    return result

# 示例用法
nums = [3, 2, 3]
print("出现次数超过n/3的元素:", majorityElement(nums))  # 输出: [3]

五、算法优势与注意事项

5.1 优势

  • 高效性:时间复杂度O(n)和空间复杂度O(1)使其成为处理大规模数据的理想选择。
  • 简洁性:算法逻辑简单,代码实现简洁,易于理解和维护。

5.2 注意事项

  • 前提条件:摩尔投票法要求数组中一定存在多数元素。如果题目没有明确这一点,必须进行验证阶段,否则可能得到错误结果。
  • 扩展性:虽然可以扩展到寻找出现次数超过n/k的元素,但随着k的增大,算法复杂度和代码实现难度也会增加。

总结

摩尔投票法是一种优雅且高效的算法,特别适合解决寻找数组中多数元素的问题,其核心思想是通过抵消不同元素的出现次数,最终找到出现次数超过一半的元素,该算法不仅时间效率高,而且空间需求极低,是处理大规模数据的有力工具。

但需要注意的是,使用该算法时必须确保题目满足存在多数元素的前提条件,否则需要进行额外的验证步骤。希望通过我在本文的描述,你能够深入理解摩尔投票法的原理和应用,并在实际编码中灵活运用这一算法。

That’s all, thanks for reading!
觉得有用就点个赞、收进收藏夹吧!关注我,获取更多干货~

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

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

相关文章

MyBatis操作数据库(2)

1.#{}和${}使用 Interger类型的参数可以看到这里显示的语句是:select username,password,age,gender,phone from userinfo where id? 输入的参数并没有在后面进行拼接,,id的值是使用?进行占位,这种sql称之为"预编译sql".这里,把#{}改成${}观察情况:这里可以看到…

C++面向对象(二)

面向对象基础内容参考&#xff1a; C面向对象&#xff08;一&#xff09;-CSDN博客 友元函数 类的友元函数是定义在类外部&#xff0c;但有权访问类的所有私有&#xff08;private&#xff09;成员和保护&#xff08;protected&#xff09;成员。尽管友元函数的原型有在类的定…

【C语言入门级教学】冒泡排序和指针数组

文章目录 1.冒泡排序2.⼆级指针3.指针数组4.指针数组模拟⼆维数组 1.冒泡排序 冒泡排序的核⼼思想&#xff1a;两两相邻的元素进⾏⽐较。 //⽅法1 void bubble_sort(int arr[], int sz)//参数接收数组元素个数 { int i 0;for(i0; i-1; i) { int j 0; for(j0; j-1; j) { …

shell脚本中常用的命令

一、设置主机名称 通过文件的方式修改通过命令修改 二、nmcli 查看网卡 ip a s ens160 (网卡名称) ifconfig ens160 nmcli device show ens160 nmcli device status nmcli connection show ens160 2.设置网卡 a)当网卡没有被设置时 b)网卡被设定&#xff0c;需要修改 三…

Nuxt3部署

最近接了一个项目&#xff0c;需要用到 nuxt3 技术来满足甲方所要求的需求&#xff0c;在部署的时候遇到了很多问题&#xff0c;这里我一一给大家讲述部署流程&#xff0c;以及所遇到的坑 打包部署 部署分为俩种方式&#xff1a; 静态(spa)部署 和 ssr部署 静态部署 静态部…

网络攻防技术一:绪论

文章目录 一、网络空间CyberSpace1、定义2、基本四要素 二、网络空间安全1、定义2、保护对象3、安全属性4、作用空间 三、网络攻击1、攻击分类2、攻击过程 四、网络防护1、定义2、安全模型3、安全服务5类4、特定安全机制8种5、普遍性安全机制5种 五、网络安全技术发展简史1、第…

【人工智能】deepseek七篇论文阅读笔记大纲

七篇文章看了整整五天&#xff0c;加上整理笔记和问ds优化&#xff0c;大致的框架是有了。具体的公式细节比较多&#xff0c;截图也比较麻烦&#xff0c;就不列入大纲去做笔记了。 DeepSeek-LLM&#xff1a;一切的起点&#xff0c;所以探索的东西比较多&#xff0c;包括&#x…

【算法】分支限界

一、基本思想 &#xff08;分支限界&#xff0c; 分枝限界&#xff0c; 分支界限 文献不同说法但都是一样的&#xff09; 分支限界法类似于回溯法&#xff0c;也是一种在问题的解空间树上搜索问题解的算法。 但一般情况下&#xff0c;分支限界法与回溯法的求解目标不同。回溯…

数据库管理与高可用-MySQL全量,增量备份与恢复

目录 #1.1MySQL数据库备份概述 1.1.1数据备份的重要性 1.1.2数据库备份类型 1.1.3常见的备份方法 #2.1数据库完全备份操作 2.1.1物理冷备份与恢复 2.1.2mysqldump备份与恢复 2.1.3MySQL增量备份与恢复 #3.1制定企业备份策略的思路 #4.1扩展&#xff1a;MySQL的GTID 4.1.1My…

从gitee仓库中恢复IDEA项目某一版本

神奇的功能&#xff01;&#xff01;&#xff01;代码改乱了&#xff0c;但是还有救&#xff01; 打开终端&#xff0c;输入git log 复制想要恢复版本的提交哈希值&#xff0c;打开终端输入git reset --hard <哈希值> &#xff0c;就能修复到那时的提交版本了

用dayjs解析时间戳,我被提了bug

引言 前几天开发中突然接到测试提的一个 Bug&#xff0c;说我的时间组件显示异常。 我很诧异&#xff0c;这里初始化数据是后端返回的&#xff0c;我什么也没改&#xff0c;这bug提给我干啥。我去问后端&#xff1a;“这数据是不是有问题&#xff1f;”。后端答&#xff1a;“…

类和对象:实现日期类

目录 概述 一.实现日期类的基本框架 二.实现比较的运算符重载 1.>的运算符重载 2.的运算符重载 3.其余的比较运算符重载 三.加减天数的运算符重载 1.,的运算符重载 2.-&#xff0c;-的运算符重载 3.对1和2的小优化 四.两个日期类相减的重载 1.&#xff0c;--的重…

基于springboot的运动员健康管理系统

博主介绍&#xff1a;java高级开发&#xff0c;从事互联网行业六年&#xff0c;熟悉各种主流语言&#xff0c;精通java、python、php、爬虫、web开发&#xff0c;已经做了六年的毕业设计程序开发&#xff0c;开发过上千套毕业设计程序&#xff0c;没有什么华丽的语言&#xff0…

华为云Flexus+DeepSeek征文 | 初探华为云ModelArts Studio:部署DeepSeek-V3/R1商用服务的详细步骤

华为云FlexusDeepSeek征文 | 初探华为云ModelArts Studio&#xff1a;部署DeepSeek-V3/R1商用服务的详细步骤 前言一、华为云ModelArts Studio平台介绍1.1 ModelArts Studio介绍1.2 ModelArts Studio主要特点1.3 ModelArts Studio使用场景1.4 ModelArts Studio产品架构 二、访问…

下载即转化的商业密码:解析华为应用商店CPD广告的智能投放逻辑

在移动互联网流量红利见顶的背景下&#xff0c;华为应用市场凭借其终端生态优势正成为开发者获客的新蓝海。数据显示&#xff0c;2025年Q1华为应用商店全球分发量同比增长27%&#xff0c;其中CPD广告因其"下载才付费"的精准特性&#xff0c;已成为金融、游戏、工具类…

分布式锁和数据库锁完成接口幂等性

1、分布式锁 唯一主键与乐观锁的本质是使用了数据库的锁&#xff0c;但由于数据库锁的性能不太好&#xff0c;所以我们可使用Redis、Zookeeper等中间件来实现分布式锁的功能&#xff0c;以Redis为例实现幂等&#xff1a;当用户通过浏览器发起请求&#xff0c;服务端接收到请求…

浅谈JMeter之常见问题Address already in use: connect

浅谈JMeter之常见问题Address already in use: connect 在JMeter高并发测试中出现“address already in use”错误&#xff0c;主要源于Windows系统的TCP端口资源耗尽及连接配置问题&#xff0c;在执行JMeter中查看结果树 原因分析 GET请求默认采用短连接&#xff08;Conne…

【机器学习基础】机器学习入门核心算法:随机森林(Random Forest)

机器学习入门核心算法&#xff1a;随机森林&#xff08;Random Forest&#xff09; 1. 算法逻辑2. 算法原理与数学推导2.1 核心组件2.2 数学推导2.3 OOB&#xff08;Out-of-Bag&#xff09;误差 3. 模型评估评估指标特征重要性可视化 4. 应用案例4.1 医疗诊断4.2 金融风控4.3 遥…

【深度学习】12. VIT与GPT 模型与语言生成:从 GPT-1 到 GPT4

VIT与GPT 模型与语言生成&#xff1a;从 GPT-1 到 GPT4 本教程将介绍 GPT 系列模型的发展历程、结构原理、训练方式以及人类反馈强化学习&#xff08;RLHF&#xff09;对生成对齐的改进。内容涵盖 GPT-1、GPT-2、GPT-3、GPT-3.5&#xff08;InstructGPT&#xff09;、ChatGPT …

常规算法学习

算法 1. 排序算法1. 归并排序1.1 普通归并排序1.2 优化后的归并排序&#xff08;TimSort&#xff09; 2. 插入排序2.1 直接插入排序2.2 二分插入排序2.3 成对插入排序 3. 快速排序3.1 单轴快速排序3.2 双轴快排 4. 计数排序 2. 树1. 红黑树&#xff08;Red Black Tree&#xff…