【问题标题】:how to use hashmap to find smallest mode in the array如何使用 hashmap 查找数组中的最小模式
【发布时间】:2016-11-16 17:03:04
【问题描述】:

问题是,给定一个数组会返回一个正整数数组中出现频率最高的元素。假设数组至少有一个元素。如果有多个模式返回最小的模式。

这是我的代码,但它不能传递这个数组 {27, 15, 15, 11, 27}。

另外,如何在 for-each 循环中不使用i 来修改我的代码?

    public int mode(int input[]){
    if(input.length==1)
        return input[0];
    Map<Integer,Integer> mymap = new HashMap<Integer,Integer>();
    int count=0;
    for(int i = 0 ;i < input.length;i++){
        count = mymap.containsKey(input[i]) ? mymap.get(input[i]) : 0;
         mymap.put(input[i],count+1);
    }
    int firstmode = -1;
    int secondmode = -1;
    int i=0;
    for(int k :mymap.keySet()){
        if(mymap.get(k)>secondmode){
            i++;
            firstmode=k;
            if(i%2==0){ //if there more than one mode,then compare them
                if(firstmode>k){
                   firstmode=k;
                   i--;
                }
            }
        secondmode=mymap.get(k);
        }  
    }
    return firstmode;
}

更新的测试用例

{1, 1, 2, 3}    should return 1 
{1, 1, 2, 3, 3} should return 1
{27, 15, 15, 11, 27}    should return 15    
{27, 15, 15, 11, 27, 27}    should return 27    
{1, 1, 6, 6, 6, 3, 3, 3}    should return 3 
{27, 15, 15, 27, 11, 11, 11, 14, 15, 15, 16, 19, 99, 100, 0, 27}    
should return 15    
{42}    should return 42

【问题讨论】:

  • 问题:mode 是什么意思?您是指最频繁出现的最小值吗?您能否提供几个示例数组,其中您的代码可以正常工作,并且上述数组的输出不正确?
  • @alwaysANewbie 。众数是一组数据中出现频率最高的值。
  • Stream.of(input).collect(Collectors.groupingBy(Function.identity(), TreeMap::new, Collectors.counting())).firstKey()
  • 是的,但是找到“最小模式”是什么意思?假设您有出现4 次的数字3,然后出现5 次的数字4。哪个是最小的?你可能会说3,因为它小于4,但3 只出现4 次,而4 出现5 次。这个问题模棱两可。

标签: java arrays algorithm dictionary


【解决方案1】:

我发现您的逻辑非常不清楚且令人费解,因此我只是从头开始重写代码。我包含了一个 Java 8 之前的解决方案和另一个 Java 8 之后的解决方案。

import java.util.HashMap;
import java.util.Map;
import java.util.IntStream;

import static java.util.Comparator.naturalOrder;

public class Solution {
    public static int smallestMode(int input[]) {
        Map<Integer, Integer> counters = new HashMap<>();
        for (Integer i : input) {
            Integer counter = counters.get(i);
            counters.put(i, counter == null ? 1 : counter + 1);
        }

        int mode = Integer.MIN_VALUE;
        for (int counter : counters.values()) mode = Math.max(mode, counter);

        int result = Integer.MAX_VALUE;
        for (Map.Entry<Integer, Integer> entry : counters.entrySet()) {
            if (entry.getValue() == mode) result = Math.min(result, entry.getKey());
        }

        return result;
    }

    public static int smallestModeJava8(int input[]) {
        Map<Integer, Long> counters = IntStream.of(input).boxed().collect(groupingBy(identity(), counting()));
        long mode = counters.values().stream().max(naturalOrder()).get();
        return counters.entrySet().stream().filter(entry -> entry.getValue() == mode).map(Map.Entry::getKey).min(naturalOrder()).get();
    }

    public static void main(String[] args) {
        int[] input = {27, 15, 15, 11, 27};
        System.out.println(smallestMode(input));
        System.out.println(smallestModeJava8(input));
    }
}

【讨论】:

  • 您的 Java 8 之前的版本仍然依赖于 Java 8 插件:Map.merge()
  • 顺便说一句,如果你有一个用于创建柜台地图的单线,我会接受它。我发现 Java 8 流对于做这么简单的事情特别难看(讨厌他们的 collector 方法)
  • @Dici,你知道如何在 Jave 7 中编写它吗?如果有多个模式,如何找到最小模式?
  • Map&lt;Integer, Integer&gt; counters = Arrays.stream(input).boxed().collect(toMap(k -&gt; k, v -&gt; 1, Integer::sum));
  • groupingBy(k -&gt; k, counting()),但那会给你Map&lt;Integer, Long&gt;
猜你喜欢
  • 2011-11-27
  • 2014-02-17
  • 2019-02-13
  • 1970-01-01
  • 1970-01-01
  • 2020-06-17
  • 1970-01-01
  • 2018-09-24
相关资源
最近更新 更多