LintCode 613. 优秀成绩

2018-02-03  本文已影响0人  Jay_8d33

原题

第一步,万年不变的查错。如果给的array是null或空,直接return 0

    public Map<Integer, Double> highFive(Record[] records) {
        Map<Integer, Double> results = new HashMap<>();
        if(records == null || records.length == 0) {
            return results;
        }
        ...
    }

这个题其实就是考PriorityQueue。实际问题简化后就是给你一堆数字,找到前五个。PriorityQueue的默认排序是从小到大,所以正好用来做这个。只留5个数字,超过5个后,每次需要加入新的,就必须要与5个数字里面最小的对比。这样就可以保证,这个queue里面的5个数就是最大的5个数。有了这个思路,剩下的就简单了,先遍历一遍array,把每个id对应的score的最高的5个用PriorityQueue的方式选出来。

        Map<Integer, Queue<Integer>> map = new HashMap<>();
        
        for (Record record : records) {
            Queue<Integer> pq = map.getOrDefault(record.id, new PriorityQueue<>());
            if (pq.size() < 5) {
                pq.add(record.score);
            } else if (record.score > pq.peek()) {
                pq.poll();
                pq.add(record.score);
            }
            map.put(record.id, pq);
        }

然后就是去平均数了。从map里面把每个queue取平均数。放到results里面,return了就完成了。

        for (Map.Entry<Integer, Queue<Integer>> entry : map.entrySet()) {
            Queue<Integer> scores = entry.getValue();
            int size = scores.size();
            int totalScore = 0;
            while (!scores.isEmpty()) {
                totalScore += scores.poll();
            }
            results.put(entry.getKey(), totalScore / (double) size);
        }
        
        return results;

完整的code

/**
 * Definition for a Record
 * class Record {
 *     public int id, score;
 *     public Record(int id, int score){
 *         this.id = id;
 *         this.score = score;
 *     }
 * }
 */
public class Solution {
    /**
     * @param records a list of <student_id, score>
     * @return find the average of 5 highest scores for each person
     * Map<Integer, Double> (student_id, average_score)
     */
    public Map<Integer, Double> highFive(Record[] records) {
        Map<Integer, Double> results = new HashMap<>();
        if (records == null || records.length == 0) {
            return results;
        } 
        
        Map<Integer, Queue<Integer>> map = new HashMap<>();
        
        for (Record record : records) {
            Queue<Integer> pq = map.getOrDefault(record.id, new PriorityQueue<>());
            if (pq.size() < 5) {
                pq.add(record.score);
            } else if (record.score > pq.peek()) {
                pq.poll();
                pq.add(record.score);
            }
            map.put(record.id, pq);
        }
        
        for (Map.Entry<Integer, Queue<Integer>> entry : map.entrySet()) {
            Queue<Integer> scores = entry.getValue();
            int size = scores.size();
            int totalScore = 0;
            while (!scores.isEmpty()) {
                totalScore += scores.poll();
            }
            results.put(entry.getKey(), totalScore / (double) size);
        }
        
        return results;
        
    }
}

分析

时间复杂度

遍历需要O(n),插入到priorityQueue需要O(n),提取因为只要前5个,所以是nlog5,所以总共就是O(n)。

空间复杂度

建立了2个map,每个size都是id的个数,所以是O(m),m是id的个数。

这个题关键在找前五,所以就是考用PriorityQueue的。没有太大的难点。

上一篇下一篇

猜你喜欢

热点阅读