LeetCode451.根据字符出现频率排序

203 阅读1分钟

🍀根据字符出现频率排序

描述:

 # 给定一个字符串 s ,根据字符出现的 频率 对其进行 降序排序 。一个字符出现的 频率 是它出现在字符串中的次数。
 ​
 返回 已排序的字符串 。如果有多个答案,返回其中任何一个。
 ​
  
 ​
 示例 1:
 ​
 输入: s = "tree"
 输出: "eert"
 解释: 'e'出现两次,'r''t'都只出现一次。
 因此'e'必须出现在'r''t'之前。此外,"eetr"也是一个有效的答案。
 示例 2:
 ​
 输入: s = "cccaaa"
 输出: "cccaaa"
 解释: 'c''a'都出现三次。此外,"aaaccc"也是有效的答案。
 注意"cacaca"是不正确的,因为相同的字母必须放在一起。
 示例 3:
 ​
 输入: s = "Aabb"
 输出: "bbAa"
 解释: 此外,"bbaA"也是一个有效的答案,但"Aabb"是不正确的。
 注意'A''a'被认为是两种不同的字符。
  
 ​
 提示:
 ​
 1 <= s.length <= 5 * 105
 s 由大小写英文字母和数字组成

思考:

这个题想了个笨方法,先记录字符串中字符出现的频率,然后再复制一个hash表遍历,每次对该字符减1,等到对应的值为0时,把它加入string字符串末尾,由于字符串加的次序是最少频率在前面,而题中要求的是最大频率在前,返回时对字符串进行reverse操作即可。

实现:

 class Solution {
     public String frequencySort(String s) {
 ​
         HashMap<Character, Integer> map = new HashMap<>();
         char[] chars = s.toCharArray();
         StringBuilder string = new StringBuilder();
 ​
         for (char c : chars) {
             if (map.containsKey(c)){
                 map.put(c, map.get(c) + 1);
             } else {
                 map.put(c, 1);
             }
         }
 ​
         HashMap<Character, Integer> record = new HashMap<>(map);
 ​
         for (int i = 0; i < chars.length; i++) {
             for (char c: record.keySet()) {
                 record.put(c, record.get(c) - 1);
                 if (record.get(c) == 0) {
                     for (int j = 0; j < map.get(c); j++) {
                         string.append(c);
                     }
                 }
             }
         }
 ​
         return string.reverse().toString();
 ​
     }
 }

测试一下!

image.png

大佬的代码:

官方的桶排序的确好的多,我写的那个太傻了。

 class Solution {
     public String frequencySort(String s) {
         Map<Character, Integer> map = new HashMap<Character, Integer>();
         int maxFreq = 0;
         int length = s.length();
         for (int i = 0; i < length; i++) {
             char c = s.charAt(i);
             int frequency = map.getOrDefault(c, 0) + 1;
             map.put(c, frequency);
             maxFreq = Math.max(maxFreq, frequency);
         }
         StringBuffer[] buckets = new StringBuffer[maxFreq + 1];
         for (int i = 0; i <= maxFreq; i++) {
             buckets[i] = new StringBuffer();
         }
         for (Map.Entry<Character, Integer> entry : map.entrySet()) {
             char c = entry.getKey();
             int frequency = entry.getValue();
             buckets[frequency].append(c);
         }
         StringBuffer sb = new StringBuffer();
         for (int i = maxFreq; i > 0; i--) {
             StringBuffer bucket = buckets[i];
             int size = bucket.length();
             for (int j = 0; j < size; j++) {
                 for (int k = 0; k < i; k++) {
                     sb.append(bucket.charAt(j));
                 }
             }
         }
         return sb.toString();
     }
 }

\