LeetCode169求众数——分治

 

 

题目:

给定一个大小为 的数组,找到其中的众数。众数是指在数组中出现次数大于 ⌊ n/2 ⌋ 的元素。

你可以假设数组是非空的,并且数组中的众数永远存在。

方法:

  • Hash Table

将每个数字仿佛哈希表中,记数,直到某个数的count>n/2。时间复杂度为o(n)。

  • 排序

因为众数是出现次数大于n/2的数字,所以排序之后中间的那个数字一定是众数。即nums[n/2]为众数。时间复杂度即排序的时间复杂度。

  • 分治法

分治法是将整个问题化简为一个一个的小问题去解,将数组分成简单的几部分,比如讲一组数分为两部分,第一部分的众数如果等于第二部分的众数,则这个数就是上一层那一组的众数,如果第一部分不等于第二部分,则遍历这一组数,记录这两个数的出现频率,返回为频率最大的,如果频率相同,返回谁都无所谓,因为在这里众数x肯定存在的,那么肯定会有至少两个x相连,如果不相连的话,那最后一个数字肯定是众数x。(例如:1 2 1 2 1 2 1,12112)。时间复杂度为o(n)。

我用了分治法解决的,下面是测试代码:

public class LC169 {
 
    static int[] nums = {1,2,1,3,1,2,1};
 
 
    public static void main(String[] args) {
 
        int result = find(nums,0,nums.length-1);
        System.out.println(result);
 
    }
 
    public static int find(int[] nums, int begin, int end){
        if (begin == end)
            return nums[begin];
        else {
            int mid = (begin+end)/2;
            int left = find(nums,begin,mid);
            int right = find(nums,mid+1,end);
 
            if(left == right)//左右两部分的众数相同 则这个数是这部分的众数
                return left;
            else{//左右两部分的众数不相同 则这两个数都有可能是这部分的众数
                //那么遍历这个数组 看一下哪个数字的出现频率高
                int countLeft = 0;
                int countRight = 0;
 
                for (int i = begin; i <= end; i++)
                    if(nums[i] == left)
                        countLeft++;
                    else if (nums[i] == right)
                        countRight++;
 
                if(countLeft>=countRight)
                    return left;
                else
                    return right;
            }
        }
    }
 
}

下面是提交代码

 

class Solution {
    public int majorityElement(int[] nums) {
        return find(nums,0,nums.length-1);        
    }
    
        public static int find(int[] nums, int begin, int end){
        if (begin == end)
            return nums[begin];
        else {
            int mid = (begin+end)/2;
            int left = find(nums,begin,mid);
            int right = find(nums,mid+1,end);
 
            if(left == right)//左右两部分的众数相同 则这个数是这部分的众数
                return left;
            else{//左右两部分的众数不相同 则这两个数都有可能是这部分的众数
                //那么遍历这个数组 看一下哪个数字的出现频率高
                int countLeft = 0;
                int countRight = 0;
                for (int i = begin; i <= end; i++)
                    if(nums[i] == left)
                        countLeft++;
                    else if (nums[i] == right)
                        countRight++;
 
                if(countLeft>countRight)
                    return left;
                else
                    return right;
            }
        }
    }
 
    
}

 

  • Moore voting alogrithm

每次从数组中找出一对不同的元素,将它们从数组中删除,直到遍历完整个数组。由于这道题已经说明一定存在一个出现次数超过一半的元素,所以遍历完数组后数组中一定会存在至少一个元素。

分治算法是一种基于分治策略的算法,它将问题分解为若干个子问题,然后递归地解决这些子问题,最后将这些子问题的解合并起来得到原问题的解。在众数问题时,可以使用分治算法来解决。 众数即为出现次数最多的数。要解一个数组中的众数,可以将数组分为两个部分,分别出左右两个部分的众数,然后再将两个众数进行比较,选出其中的众数分治算法众数的步骤如下: 1. 将数组分为左右两个部分,分别递归解左右两个部分的众数。 2. 如果左右两个部分的众数相同,则直接返回这个众数。 3. 如果左右两个部分的众数不同,则统计左右两个部分中各自的众数出现的次数,选出次数更多的那个数作为整个数组的众数。 Java代码实现如下: ```java public int majorityElement(int[] nums) { if (nums.length == 1) { return nums[0]; } int mid = nums.length / 2; int[] left = Arrays.copyOfRange(nums, 0, mid); int[] right = Arrays.copyOfRange(nums, mid, nums.length); int leftMajority = majorityElement(left); int rightMajority = majorityElement(right); if (leftMajority == rightMajority) { return leftMajority; } int leftCount = count(nums, leftMajority); int rightCount = count(nums, rightMajority); return leftCount > rightCount ? leftMajority : rightMajority; } private int count(int[] nums, int num) { int count = 0; for (int i = 0; i < nums.length; i++) { if (nums[i] == num) { count++; } } return count; } ``` 这样,就可以使用分治算法来解决众数问题了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值