题目
数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。
你可以假设数组是非空的,并且给定的数组总是存在多数元素。
示例 1:
输入: [1, 2, 3, 2, 2, 2, 5, 4, 2]
输出: 2
限制:
1 <= 数组长度 <= 50000
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/shu-zu-zhong-chu-xian-ci-shu-chao-guo-yi-ban-de-shu-zi-lcof
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
代码
方法一
首先是最容易想到的遍历计数法
使用HashMap来计数(只需要注意一下hashMap的getOrDefault方法的使用
之后遍历HashMap求最大值就可以了
只不过时间复杂度感人 O(N^2)
public int majorityElement(int[] nums) {
HashMap<Integer, Integer> hashMap = new HashMap<>();
for (int i = 0; i < nums.length; i++) {
hashMap.put(nums[i] , hashMap.getOrDefault(i, 0) + 1);
}
for (int i: hashMap.keySet()
) {
System.out.println(hashMap.get(i));
if(hashMap.get(i) >= nums.length / 2){
return nums[i];
}
}
return 0;
}
}
方法二
接下来就是利用题目特性(数字超过一半
那么数组中间的那个数就一定是出现次数超过一半的数
时间复杂度O(nlogN) (排序算法的时间复杂度
class Solution {
public int majorityElement(int[] nums) {
Arrays.sort(nums);
return nums[nums.length / 2];
}
}
方法三
摩尔投票法((((这谁知道啊, 为什么要这样取名啊
((说的阳间一点就是票数正负抵消
即: 目标数字的个数 > 其它数字的个数
核心就是: 对拼消耗(我血比你多输出和你一样那我疯狂和你换血, The Shy行为(x
时间复杂度为O(N)
我们把候选人定为temp = nums[0] (出生自带一票 count = 1
当遇到与temp相同的元素, 则让temp + 1 , 否则让temp - 1
当temp = 0时, 更换候选人
class Solution {
public int majorityElement(int[] nums) {
int temp = nums[0];
int count = 0;
for (int num: nums
) {
if (num == temp) {
count++;
}else {
count--;
}
if (count == 0) {
temp = num;
count = 1;
}
}
return temp;
}
}