🍀根据字符出现频率排序
描述:
# 给定一个字符串 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();
}
}
测试一下!
大佬的代码:
官方的桶排序的确好的多,我写的那个太傻了。
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();
}
}
\