【LeetCode】692. 前K个高频单词

这篇具有很好参考价值的文章主要介绍了【LeetCode】692. 前K个高频单词。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

描述

给定一个单词列表 words 和一个整数 k ,返回前 k 个出现次数最多的单词。

返回的答案应该按单词出现频率由高到低排序。如果不同的单词有相同出现频率, 按字典顺序 排序

示例

示例1

输入: words = [“i”, “love”, “leetcode”, “i”, “love”, “coding”], k = 2
输出: [“i”, “love”]
解析: “i” 和 “love” 为出现次数最多的两个单词,均为2次。
注意,按字母顺序 “i” 在 “love” 之前。

示例2
输入: [“the”, “day”, “is”, “sunny”, “the”, “the”, “the”, “sunny”, “is”, “is”], k = 4
输出: [“the”, “is”, “sunny”, “day”]
解析: “the”, “is”, “sunny” 和 “day” 是出现次数最多的四个单词,出现次数依次为 4, 3, 2 和 1 次。

解题思路及事项

思路一

遇到这样的题,我们一般思路肯定就是TOP-K问题,这样想当然没有问题,但是我们这里数据没那么多,用到这里属于杀鸡焉用牛刀,不过我们可以试一试,等下在讲别的思路。

不管是那个思路,首先这是一对一的关系,我们肯定要先用到map,,统计不同字符串出现的次数。

TOP-K在于建大堆和小堆的问题,这道题建议建大堆。我们现在已经学了,C++,因此可以使用priority_queue它默认就是建大堆,
【LeetCode】692. 前K个高频单词,LeetCood,c++,leetcode,算法
然后把前K个元素拿出来就好了

class Solution {
public:
    vector<string> topKFrequent(vector<string>& words, int k) {
        map<string,int> mp;
        for(auto& str:words)
        {
            mp[str]++;
        }

        vector<string> ret;
        //这里我们建一个大堆
        priority_queue<pair<string,int>>> py;
        auto it=mp.begin();
        while(it != mp.end())
        {
            py.push(*it);
            ++it;
        }

        while(k--)
        {
            ret.push_back(py.top().first);
            py.pop();
        }

        return ret;
    }
};

这是根据我们的思路写出来的代码
【LeetCode】692. 前K个高频单词,LeetCood,c++,leetcode,算法
但是结果不对,难道我们思路出现了问题,这道题不应该这样解,
其实并不是,这样的思路是对的,但是问题就在于priority_queue第三个参数仿函数的比较出现了问题。
因为它比较的是pair对象。而pair的相关比较函数我们可以看看到底是怎么比的
【LeetCode】692. 前K个高频单词,LeetCood,c++,leetcode,算法
可以看到pair是先比较first,如果first相等在比较second。
但是我们的pair第一个参数是string,第二个参数是int。
这于我们想要优先比较int就不对,因此我们自己写一个仿函数。

class Solution {
public:

    template<class T>
    struct Less
    {
        bool operator()(const pair<string,int>& l,const pair<string,int>& r)
        {
            return l.second < r.second;
        }
    };

    vector<string> topKFrequent(vector<string>& words, int k) {
        map<string,int> mp;
        for(auto& str:words)
        {
            mp[str]++;
        }

        vector<string> ret;
        //这里我们建一个大堆
        priority_queue<pair<string,int>,vector<pair<string,int>>,Less<pair<string,int>>> py;
        auto it=mp.begin();
        while(it != mp.end())
        {
            py.push(*it);
            ++it;
        }

        while(k--)
        {
            ret.push_back(py.top().first);
            py.pop();
        }

        return ret;


    }
};

【LeetCode】692. 前K个高频单词,LeetCood,c++,leetcode,算法
运行结果还是出现了问题。经过分析可能是建大堆出现了问题,我们打印一下看看是不是这个问题。
【LeetCode】692. 前K个高频单词,LeetCood,c++,leetcode,算法
经过对比发现,它们出现次数都是6次,就是建立大堆谁在上面谁在下面出现了问题。

注意看到我们的题目要求,不同单词出现相同频率,按 字典顺序 排序
【LeetCode】692. 前K个高频单词,LeetCood,c++,leetcode,算法
而我们在写自己的仿函数的时候,只考虑了出现次数不同的情况,而没有考虑这个情况。

class Solution {
public:

    template<class T>
    struct Less
    {
        bool operator()(const pair<string,int>& l,const pair<string,int>& r)
        {
        //出现次数相同,就按 字典顺序 排序
            return l.second < r.second || (l.second == r.second && l.first > r.first);
        }
    };

    vector<string> topKFrequent(vector<string>& words, int k) {
        map<string,int> mp;
        for(auto& str:words)
        {
            mp[str]++;
        }


        // for(auto& e: mp)
        // {
        //     cout<<e.first<<":"<<e.second<<endl;
        // }
        vector<string> ret;
        //这里我们建一个大堆
        priority_queue<pair<string,int>,vector<pair<string,int>>,Less<pair<string,int>>> py;
        auto it=mp.begin();
        while(it != mp.end())
        {
            py.push(*it);
            ++it;
        }

        while(k--)
        {
            ret.push_back(py.top().first);
            py.pop();
        }

        return ret;       
    }
};

思路二

刚才说过使用堆来对少的数据排序,杀鸡焉用牛刀了。现在想一想我用map建立一对一的关系之后,我给它排序一下不就好了吗,反正有算法库给我提供的sort函数。那来试一试

注意sort底层使用的快速排序,结构是线性结构,而map并不是线性结构而是树形结构,因此要把map里的数据放在vector,才能使用sort。
sort默认是升序,第一个版本是按照运算符<比较的,第二个版本是按照comp比较的也就是说我们给它提供一个仿函数按照自己的想法比较。
【LeetCode】692. 前K个高频单词,LeetCood,c++,leetcode,算法
由TOP-K我们就知道如果直接让pair对比会有问题,所以我们选第二种。

class Solution {
public:
    struct Compare
    {
        bool operator()(const pair<string,int>& l,const pair<string,int>& r)
        {
            return l.second > r.second || (l.second == r.second && l.first < r.first);
        }
    };

    vector<string> topKFrequent(vector<string>& words, int k) {
        map<string,int> mp;
        for(auto& str:words)
        {
            mp[str]++;
        }
        
        vector<string> ret;
        vector<pair<string,int>> v;
        for(auto& e:mp)
        {
            v.push_back(e);
        }
		//这个Compare我们是按照降序进行判断的
        sort(v.begin(),v.end(),Compare());

        for(int i=0;i<k;++i)
        {
            ret.push_back(v[i].first);
        }

        return ret;
   }
};

这样也能解决问题,不过这样的sort并不能保持稳定性,需要我自己手动控制才能保持稳定性以达到相同次数按 字典顺序 排序。
下面介绍一种稳定的排序算法。
【LeetCode】692. 前K个高频单词,LeetCood,c++,leetcode,算法
stable_sort,可以保持排序的稳定性。
【LeetCode】692. 前K个高频单词,LeetCood,c++,leetcode,算法
i 在 love的前面,出现次数相同,i 依旧在 love前面。文章来源地址https://www.toymoban.com/news/detail-767574.html

class Solution {
public:
    struct Compare
    {
        bool operator()(const pair<string,int>& l,const pair<string,int>& r)
        {
            return l.second > r.second ;
        }
    };

    vector<string> topKFrequent(vector<string>& words, int k) {
        map<string,int> mp;
        for(auto& str:words)
        {
            mp[str]++;
        }
        
        vector<string> ret;
        vector<pair<string,int>> v;
        for(auto& e:mp)
        {
            v.push_back(e);
        }
		//这个Compare我们是按照降序进行判断的
        //sort(v.begin(),v.end(),Compare());
        stable_sort(v.begin(),v.end(),Compare());

        for(int i=0;i<k;++i)
        {
            ret.push_back(v[i].first);
        }

        return ret;
   }
};

到了这里,关于【LeetCode】692. 前K个高频单词的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 高频算法题冒险之旅精讲(一)之LeetCode小牛试刀五道题

    📢 导读: 本篇博文是LeetCode算法题讲解篇,对高频算法题进行详细而深入的讲解,解题语言选择的是Java。 更多算法专栏如下: ⛳️ 排序算法 ⛳️ 分治法 ⛳️ LeetCode高频算法题讲解 ⛳️ 数据结构 前言: 本次算法冒险之旅将围绕LeetCode上面的算法面试题汇总进行讲解,该

    2024年02月02日
    浏览(38)
  • 算法学习——LeetCode力扣图论篇3(127. 单词接龙、463. 岛屿的周长、684. 冗余连接、685. 冗余连接 II)

    127. 单词接龙 - 力扣(LeetCode) 描述 字典 wordList 中从单词 beginWord 和 endWord 的 转换序列 是一个按下述规格形成的序列 beginWord - s1 - s2 - … - sk: 每一对相邻的单词只差一个字母。 对于 1 = i = k 时,每个 si 都在 wordList 中。注意, beginWord 不需要在 wordList 中。 sk == endWord 给你两

    2024年04月09日
    浏览(77)
  • 数据结构与算法之字符串: Leetcode 557. 反转字符串中的单词 III (Typescript版)

    翻转字符串中的单词 III https://leetcode.cn/problems/reverse-words-in-a-string-iii/ 描述 给定一个字符串 s ,你需要反转字符串中每个单词的字符顺序,同时仍保留空格和单词的初始顺序。 示例 1: 示例 2: 提示: 1 = s.length = 5 * 1 0 4 10^4 1 0 4 s 包含可打印的 ASCII 字符。 s 不包含任何开头或

    2024年02月01日
    浏览(77)
  • Leetcode 139.单词拆分

    OJ链接 :139.单词拆分  代码:    

    2024年02月04日
    浏览(41)
  • 【LeetCode】79.单词搜索

    给定一个  m x n  二维字符网格  board  和一个字符串单词  word  。如果  word  存在于网格中,返回  true  ;否则,返回  false  。 单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格。同一个单元格内的字母不

    2024年02月15日
    浏览(43)
  • leetcode 动态规划(单词拆分)

    139.单词拆分 力扣题目链接(opens new window) 给定一个非空字符串 s 和一个包含非空单词的列表 wordDict,判定 s 是否可以被空格拆分为一个或多个在字典中出现的单词。 说明: 拆分时可以重复使用字典中的单词。 你可以假设字典中没有重复的单词。 示例 1: 输入: s = “leetcode”

    2024年02月22日
    浏览(41)
  • leetcode单词的个数

    题目描述 解题思路 执行结果 leetcode 题目描述 统计字符串中的单词个数,这里的单词指的是连续的不是空格的字符。 请注意,你可以假定字符串里不包括任何不可打印的字符。 示例: 输入: \\\"Hello, my name is John\\\" 输出: 5 解释: 这里的单词是指连续的不是空格的字符,所以 \\\"Hello

    2023年04月13日
    浏览(26)
  • Leetcode 最后一个单词的长度

    给你一个字符串 s,由若干单词组成,单词前后用一些空格字符隔开。返回字符串中 最后一个 单词的长度。 单词 是指仅由字母组成、不包含任何空格字符的最大子字符串。 示例 1: 输入:s = “Hello World” 输出:5 解释:最后一个单词是“World”,长度为5。 示例 2: JavaScr

    2024年02月10日
    浏览(42)
  • 【LeetCode-中等题】79. 单词搜索

    需要一个标记数组 来标志格子字符是否被使用过了 先找到word 的第一个字符在表格中的位置,再开始递归 递归的结束条件是如果word递归到了最后一个字符了,说明能在矩阵中找到单词 剪枝条件 就是如果已经找到单词了 res = true 了 后面就不需要递归了,还有如果下标越界、

    2024年02月09日
    浏览(39)
  • 58.leetcode 最后一个单词的长度

    分2种情况 第一种情况只有一个单词,不包含空格:这种情况直接返回单词本身的长度。 第二种情况包含空格:先去掉首尾的空格,根据空格切割字符串生成一个字符串列表,返回倒数第一个索引位置字符串的长度

    2024年02月01日
    浏览(35)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包