[剑指 offer]39 数组中出现次数超过一半的数字 三种解法 Java实现

这篇博客介绍了如何在数组中找到出现次数超过一半的数字,提供了三种不同的解决方案:遍历计数、排序查找和摩尔投票法。时间复杂度分别为O(N^2)、O(nlogN)和O(N)。对于大型数组,摩尔投票法更为高效。
摘要由CSDN通过智能技术生成

在这里插入图片描述

题目

数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。

你可以假设数组是非空的,并且给定的数组总是存在多数元素。

示例 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;
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值