算法剖析1:摩尔投票算法 ——寻找出现次数超过一半的数

一、算法概述

在算法面试和刷题中,我们经常会遇到这样一类问题:在一个数组中找到出现次数超过一半的元素。如果用暴力统计或者哈希表的方法,虽然能解决问题,但要么时间复杂度高,要么空间复杂度不够理想。今天我要给大家介绍的摩尔投票法(Boyer-Moore Majority Vote Algorithm),正是解决这类问题的 "最优解",它可以在 O(N) 时间复杂度和 O(1) 空间复杂度内找到答案。

二、问题背景

现在给你一个长度为n的数组,有一个数出现次数最多,请你设计一个算法找出这个数。

这道题的核心就是 "找多数元素"------ 即出现次数最多的元素唯一。题目保证这样的元素一定存在,这也是摩尔投票法能够生效的前提条件。

方案一:暴力统计

数据量大时,容易超时

方案二;哈希法

空间开销大

方案三:摩尔投票法

完美解决

三、算法理解

想象一下,你现在来到一个新的班级,年级主任要求你们班必须得有一个班长,并且现在就得选出来。而你作为班主任,在不知道谁更适合的情况下,你会怎么做呢?你会这样做,从序号为1同学开始,让每个人开始当班长。按以下规则筛选:

令 临时班长=张三 可靠度=1(默认选择相信他)

1.如果当前班长没有可靠度,换人 //滚蛋

2.如果临时班长被人投诉,可靠度-1;//支持

3.如果临时班长干的还不错,表扬一波,可靠度+1;//不支持

四、算法原理

  • 由于题目保证存在一个数字出现次数最多,我们可以通过 "抵消" 的思想来找到这个数。

  • 维护一个候选数 candidate 和一个计数器 count

  • 遍历矩阵中的每个元素:

    • 如果 count == 0,就将当前元素设为候选数。

    • 如果当前元素等于候选数,count++;否则,count--

  • 最终剩下的候选数就是出现次数超过一半的数。

五、算法实现

令 currNum=?(暂时不知道),count=0

遍历数组:(int a[n])

(假设当前数为a[i])

if(count==0)

{

currNum=a[i];

count++;//默认投支持票

}

else if(currNum==a[i])count++;

else if(currNum!=a[i])count--;

这样们就可以找到出现次数最多的元素了

六、真题实解

代码

cpp 复制代码
#include<bits/stdc++.h>
using namespace std; 

int main()
{
	ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
	int n,m;
	cin>>n>>m;
	int count=0;
	int wait;
	for(int i=0;i<n;i++)for(int j=0;j<m;j++)
	{
		int x;
		cin>>x;
		if(count==0)
    {
      wait=x;
      count++;
    }
		else if(wait==x)count++;
		else if(wait!=x)count--;
	}
	cout<<wait<<endl;
	return 0;
}
相关推荐
lightqjx4 分钟前
【C++】unordered系列的封装
开发语言·c++·stl·unordered系列
天天爱吃肉821818 分钟前
跟着创意天才周杰伦学新能源汽车研发测试!3年从工程师到领域专家的成长秘籍!
数据库·python·算法·分类·汽车
alphaTao29 分钟前
LeetCode 每日一题 2026/2/2-2026/2/8
算法·leetcode
甄心爱学习33 分钟前
【leetcode】判断平衡二叉树
python·算法·leetcode
阿猿收手吧!37 分钟前
【C++】string_view:高效字符串处理指南
开发语言·c++
颜酱42 分钟前
从二叉树到衍生结构:5种高频树结构原理+解析
javascript·后端·算法
不知名XL1 小时前
day50 单调栈
数据结构·算法·leetcode
Word码1 小时前
[C++语法] 继承 (用法详解)
java·jvm·c++
@––––––1 小时前
力扣hot100—系列2-多维动态规划
算法·leetcode·动态规划
lxl13071 小时前
C++算法(1)双指针
开发语言·c++