题解 | #字符串出现次数的TopK问题#

字符串出现次数的TopK问题

http://www.nowcoder.com/practice/fd711bdfa0e840b381d7e1b82183b3ee

思路

这个题目要求时间复杂度,想到用优先队列,超过k个可以出队列,保证了调整堆的时间为,再加上外层循环遍历一遍数组,时间复杂度正好满足要求。
要统计出现字符串的频率,我们可以用一个哈希表记录。
排序要实现按字符串出现频率由高到低排序。如果不同的字符串有相同出现频率,按字典序排序。那么我们得自定义排序,根据数字我们制作小顶堆,因为只有大的在下面,到时元素个数大于k是,可以把出现频率小的出队列。
对于频率相同的,根据字典排序。

代码

import java.util.*;


public class Solution {
    /**
     * return topK string
     * @param strings string字符串一维数组 strings
     * @param k int整型 the k
     * @return string字符串二维数组
     */
    // 利用java的内部类定义一个节点,用于记录字符串及其出现的次数
    public static class Node{
        private String str; // 记录字符串
        private Integer count; // 记录字符串出现的次数
        public Node(String _str, Integer _count){ // 节点的有参构造
            this.str = _str;
            this.count = _count;
        }
    }
    public String[][] topKstrings (String[] strings, int k) {
        // 自定义排序
        PriorityQueue<Node> queue = new PriorityQueue<>(new Comparator<Node>(){
            @Override
            public int compare(Node o1, Node o2){
                // java的优先队列底层就是小根堆,次数大的在下面
                if (o1.count > o2.count){
                    return 1;
                } else if (o1.count < o2.count){
                    return -1;
                } else {
                    // 频率相同,字典排序前要在下面
                    return o2.str.compareTo(o1.str);
                }
            }
        });
        HashMap<String, Integer> map = new HashMap<>();
        // 哈希表记录字符串及其频率
        for (String s : strings){
            map.put(s,map.getOrDefault(s,0)+1);
        }
        // 遍历哈希表
        for (Map.Entry<String, Integer> entry : map.entrySet()){
            String key = entry.getKey();
            Integer value = entry.getValue();
            queue.offer(new Node(key, value));
            // 队列中的元素大于k,及时出队列
            if (queue.size() > k){
                queue.poll();
            }
        }
        String[][] result = new String[k][2];
        // 遍历队列,加入结果数组
        for (int i = k - 1; i >= 0; i--){
            Node node = queue.poll();
            result[i][0] = node.str;
            result[i][1] = Integer.toString(node.count);
        }
        return result;
    }
}
全部评论

相关推荐

点赞 收藏 评论
分享
正在热议
# 牛客帮帮团来啦!有问必答 #
1151650次浏览 17149人参与
# 通信和硬件还有转码的必要吗 #
11203次浏览 101人参与
# OPPO开奖 #
19203次浏览 267人参与
# 和牛牛一起刷题打卡 #
18978次浏览 1635人参与
# 实习与准备秋招该如何平衡 #
203393次浏览 3627人参与
# 大厂无回复,继续等待还是奔赴小厂 #
4972次浏览 30人参与
# 不去互联网可以去金融科技 #
20390次浏览 255人参与
# 通信硬件薪资爆料 #
265915次浏览 2484人参与
# 国企是理工四大天坑的最好选择吗 #
2227次浏览 34人参与
# 互联网公司评价 #
97688次浏览 1280人参与
# 简历无回复,你会继续海投还是优化再投? #
25037次浏览 354人参与
# 0offer是寒冬太冷还是我太菜 #
454871次浏览 5124人参与
# 国企和大厂硬件兄弟怎么选? #
53903次浏览 1012人参与
# 参加过提前批的机械人,你们还参加秋招么 #
14645次浏览 349人参与
# 硬件人的简历怎么写 #
82286次浏览 852人参与
# 面试被问第一学历差时该怎么回答 #
19398次浏览 213人参与
# 你见过最离谱的招聘要求是什么? #
28103次浏览 248人参与
# 学历对求职的影响 #
161242次浏览 1804人参与
# 你收到了团子的OC了吗 #
538745次浏览 6387人参与
# 你已经投递多少份简历了 #
344237次浏览 4963人参与
# 实习生应该准时下班吗 #
96977次浏览 722人参与
# 听劝,我这个简历该怎么改? #
63525次浏览 622人参与
牛客网
牛客企业服务