简介
题目链接:https://leetcode.cn/problems/group-anagrams/description/
解决方式:数组 + 哈希表
这是作者学习众多大神的思路进行解题的步骤,很推荐大家解题的时候去看看题解里面大佬们的思路、想法!
哈希表
排序
思路:字母异位词,也就是字符种类与个数相同,不过排序不一样。那么我们就可以迭代数组中的每个字符串,将其按26字母表的顺序重新排序,只要是字母异位词,排序的结果就相同。我们可以将排序后的结果作为哈希表的 key,集合作为哈希表的 value。这样,字母异位词被映射到相同的 key,就可以存储进同一个分组的集合中了。
java
class Solution {
public List<List<String>> groupAnagrams(String[] strs) {
// 哈希表
Map<String,List<String>> map = new HashMap<String,List<String>>();
// 迭代数组
for (String str : strs){
// 字符串转为字符数组
char[] array = str.toCharArray();
// 重新排序,排序结果作为字母异位词的 key
Arrays.sort(array);
// 转为 String
String s = new String(array);
// 从哈希表中拿到指定 key 的 value,没有就创建
List<String> list = map.getOrDefault(s, new ArrayList<String>());
list.add(str);
// 更新 key 对应的 value
map.put(s,list);
}
// 返回结果
return new ArrayList<List<String>>(map.values());
}
}
计数
思路:大体逻辑与上面的方法相同,都是经过观察,构建唯一的 key,通过这个 key 进行分组。只不过计数这个方法构建唯一 key 是通过统计每个字符出现的次数,然后拼接成字符串来实现的。
java
class Solution {
public List<List<String>> groupAnagrams(String[] strs) {
// 哈希表
Map<String,List<String>> map = new HashMap<String,List<String>>();
for (String str : strs){
int[] count = new int[26];
int length = str.length();
// 记录字母出现的次数
for(int i = 0; i < length; i++){
// 隐式含有字母排序的意思
// 因为 count 数组索引对应的是 a-z
count[str.charAt(i) - 'a']++;
}
// 构建唯一的键
// 将每个出现次数大于 0 的字母和出现次数按顺序拼接成字符串,作为哈希表的键
StringBuilder sb = new StringBuilder();
for(int i = 0; i < 26; i++){
if(count[i] != 0){
// 索引转为字母
sb.append((char)(i + 'a'));
sb.append(count[i]);
}
}
String key = sb.toString();
// 获取键对应的值
List<String> list = map.getOrDefault(key, new ArrayList<String>());
list.add(str);
map.put(key, list);
}
return new ArrayList<List<String>>(map.values());
}
}