leetcode 451.根据字符出现频率排序

文章目录

LeetCode第451题"根据字符出现频率排序"要求根据字符串中字符的出现频率对字符串进行排序。可以通过以下步骤解决这个问题:

方法一

  1. 使用哈希表统计每个字符的频率。
  2. 使用最大堆根据频率对字符进行排序。
  3. 根据排序结果构造输出字符串。

下面是C++实现的具体代码:

代码实现

cpp 复制代码
#include <iostream>
#include <string>
#include <unordered_map>
#include <vector>
#include <queue>
#include <utility>
using namespace std;

class Solution {
public:
    string frequencySort(string s) {
        // 1. 统计每个字符的频率
        unordered_map<char, int> frequencyMap;
        for (char c : s) {
            frequencyMap[c]++;
        }

        // 2. 使用最大堆根据频率排序字符
        priority_queue<pair<int, char>> maxHeap;
        for (auto& entry : frequencyMap) {
            maxHeap.push({entry.second, entry.first});
        }

        // 3. 根据排序结果构造输出字符串
        string result;
        while (!maxHeap.empty()) {
            auto [freq, ch] = maxHeap.top();
            maxHeap.pop();
            result.append(freq, ch);
        }

        return result;
    }
};

int main() {
    Solution solution;
    string s = "tree";
    string result = solution.frequencySort(s);
    cout << "Sorted by frequency: " << result << endl;
    return 0;
}

解释

1. 统计每个字符的频率
cpp 复制代码
unordered_map<char, int> frequencyMap;
for (char c : s) {
    frequencyMap[c]++;
}
  • 使用哈希表frequencyMap统计字符串中每个字符的出现频率。
2. 使用最大堆根据频率排序字符
cpp 复制代码
priority_queue<pair<int, char>> maxHeap;
for (auto& entry : frequencyMap) {
    maxHeap.push({entry.second, entry.first});
}
  • 使用优先队列maxHeap,其中元素是pair<int, char>,表示字符及其频率。
  • 将每个字符及其频率插入最大堆中。
3. 根据排序结果构造输出字符串
cpp 复制代码
string result;
while (!maxHeap.empty()) {
    auto [freq, ch] = maxHeap.top();
    maxHeap.pop();
    result.append(freq, ch);
}
  • 从最大堆中依次取出字符及其频率,将字符按频率重复添加到结果字符串中。

总结

这个方法的时间复杂度主要是O(N log N),其中N是字符串的长度。通过这种方式,可以高效地根据字符出现的频率对字符串进行排序。

方法二

这个实现方法主要通过哈希表统计字符频率,然后使用自定义排序规则对字符串进行排序。具体步骤如下:

  1. 使用哈希表统计每个字符的出现频率。
  2. 定义一个比较函数,根据字符的频率和字典序对字符进行排序。
  3. 使用 std::sort 函数对字符串进行排序。

下面是对这段代码的详细解释:

cpp 复制代码
#include <string>
#include <unordered_map>
#include <algorithm>
using std::unordered_map;
using std::string;
/*
 * @lc app=leetcode.cn id=451 lang=cpp
 *
 * [451] 根据字符出现频率排序
 */

// @lc code=start
class Solution {
public:
    string frequencySort(string s) {
        // 1. 使用哈希表统计每个字符的出现频率
        unordered_map<char, int> dict;  // 哈希表,键为字符,值为该字符的出现频率
        for (auto c : s) {
            dict[c]++;
        }

        // 2. 定义一个比较函数
        auto compare = [&dict](const char& a, const char& b) {
            return (dict[a] > dict[b]) || (dict[a] == dict[b] && (a < b));
        };

        // 3. 使用 std::sort 函数对字符串进行排序
        std::sort(s.begin(), s.end(), compare);

        return s;
    }
};
// @lc code=end

详细解释

  1. 统计字符频率

    cpp 复制代码
    unordered_map<char, int> dict;
    for (auto c : s) {
        dict[c]++;
    }
    • dict 是一个 unordered_map,用于存储每个字符及其出现的频率。
    • 遍历字符串 s,将每个字符的频率记录到 dict 中。
  2. 定义比较函数

    cpp 复制代码
    auto compare = [&dict](const char& a, const char& b) {
        return (dict[a] > dict[b]) || (dict[a] == dict[b] && (a < b));
    };
    • compare 是一个 lambda 表达式,用于自定义排序规则。
    • dict[a] > dict[b]:按字符频率从高到低排序。
    • dict[a] == dict[b] && (a < b):如果频率相同,则按字符的字典序从小到大排序。
  3. 排序字符串

    cpp 复制代码
    std::sort(s.begin(), s.end(), compare);
    • 使用 std::sort 函数对字符串 s 进行排序。
    • s.begin()s.end() 分别表示字符串的起始迭代器和结束迭代器。
    • compare 函数作为自定义的比较规则。

示例

假设输入字符串为 "tree"

  1. 统计字符频率:

    t: 1
    r: 1
    e: 2
    
  2. 自定义比较函数会将字符按频率从高到低排序,如果频率相同,则按字典序排序。

  3. 排序后的字符串为 "eert"

运行效果

这段代码可以正确地根据字符出现的频率对字符串进行排序,满足题目要求。你可以在你的环境中运行这个代码来验证其正确性。

相关推荐
KeyPan1 小时前
【视觉SLAM:八、后端Ⅰ】
人工智能·数码相机·算法·机器学习·计算机视觉
Jackilina_Stone1 小时前
【论文阅读笔记】SCI算法与代码 | 低照度图像增强 | 2022.4.21
论文阅读·人工智能·笔记·python·算法·计算机视觉
sysu633 小时前
59.螺旋矩阵Ⅱ python
数据结构·python·算法·leetcode·面试
星空_MAX3 小时前
C语言优化技巧--达夫设备(Duff‘s Device)解析
c语言·数据结构·c++·算法
Wang's Blog3 小时前
数据结构与算法之动态规划: LeetCode 62. 不同路径 (Ts版)
算法·leetcode·动态规划
Pipi_Xia_4 小时前
相切于球体上定点的平面
线性代数·算法·平面·几何学
Wang's Blog4 小时前
数据结构与算法之动态规划: LeetCode 72. 编辑距离 (Ts版)
算法·leetcode·动态规划
m0_694938015 小时前
Leetcode打卡:切蛋糕的最小总开销II
算法·leetcode·职场和发展
Pipi_Xia_5 小时前
过圆外一点与圆相切的直线
线性代数·算法·平面·几何学