剑指offer_28
题目描述
数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}。由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2。如果不存在则输出0。
题解
这道题,最直白的解法是使用一个 map 来记录各个数字出现的次数,最后取出现次数最多的作为解。但这个方法需要消耗额外的空间,不是最优。下面是最优解法:
我们做这样的想象,现在有来自不同阵营的多支部队,他们互为敌人。每个士兵都容不得敌人,宁愿与敌人同归于尽。可以想象,如果某个阵营的士兵数量超过所有阵营士兵总数的一半,该阵营士兵一换一带走一个其他阵营的,最终剩下的就是该阵营的士兵了,该阵营就获胜了。
这和本题有什么关系呢?且看下面的故事:
现在要打仗了,所有士兵依次进入战场,如果战场上有其他阵营的士兵,他就与其中一个同归于尽。否则,他就留在战场上。容易想象,战场上要么没有人,要么是同一阵营的。所有士兵都进入战场后,最终战场上剩下的只会是同一阵营的,而该阵营就是人数过半的那个。
我们要模拟这个过程,找到那个获胜的阵营。数组中每个元素就是士兵,元素的值,就是这个士兵所属阵营。
java代码
public class Solution {
public int MoreThanHalfNum_Solution(int [] array) {
int count = 0;
int win = array[0];
for(int i = 0; i < array.length; i++){
if(count == 0){
win = array[i];
count = 1;
}else{
//同一个数字
if(win == array[i]) count++;
else count--;
}
}
int sum = 0;
//判断win的个数是否大于数组长度的一半
for(int i = 0; i < array.length; i++){
if(array[i] == win) sum++;
}
System.out.println("sum:" + sum);
if(sum > array.length/2) return win;
else return 0;
}
}