题目描述
数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。例如输入一个长度为9
的数组{1,2,3,2,2,2,5,4,2}
。由于数字2
在数组中出现了5
次,超过数组长度的一半,因此输出2
。如果不存在则输出0
。
示例1
输入:[1,2,3,2,2,2,5,4,2]
输出:2
思路 & 解答
- 如果使用
hashmap
直接统计,需要额外的空间,我们不希望使用额外空间; - 如果使用排序之后再统计,需要时间复杂度为O(nlogn),我们希望时间复杂度更低一点。
于是我们使用了多数投票法,先取第一个数字,出现次数是1
,遍历后面的数字,如果相同则加1,如果不同则减1
,结果如果为0
,则把当前的数字更新,而且次数更新为1
。
直到最后,则可以获取到出现次数最多的数字,再次遍历数组,统计该数字出现的次数,如果次数大于长度的一半,则说明是该数字。
代码如下:
public class Solution {
public int MoreThanHalfNum_Solution(int [] array) {
if(array==null||array.length==0){
return 0;
}else{
int num=array[0];
int count=1;
for(int i=1;i<array.length;i++){
if(array[i]==num){
count++;
}else{
count--;
}
if(count==0){
num=array[i];
count=1;
}
}
count=0;
for(int i=0;i<array.length;i++){
if(array[i]==num){
count++;
}
}
if(count>array.length/2){
return num;
}else{
return 0;
}
}
}
}
C++
代码如下:
class Solution {
public:
int MoreThanHalfNum_Solution(vector<int> array) {
if (array.size() == 0) {
return 0;
} else {
int num = array[0];
int count = 1;
for (int i = 1; i < array.size(); i++) {
if (array[i] == num) {
count++;
} else {
count--;
}
if (count == 0) {
num = array[i];
count = 1;
}
}
count = 0;
for (int i = 0; i < array.size(); i++) {
if (array[i] == num) {
count++;
}
}
if (count > array.size() / 2) {
return num;
} else {
return 0;
}
}
}
};
- 时间复杂度:O(n)
- 空间复杂度:O(1)