LeetCode 49 字母异位词分组

点此看全部题解 LeetCode必刷100题:一份来自面试官的算法地图(题解持续更新中)

生活中的算法

你有没有玩过扑克牌?打完一局之后,我们通常会把散落的牌收集起来,按照花色分组整理。虽然每张牌的花色图案位置可能不同,但只要是同一个花色,我们就把它们放在一起。

这和我们今天要讲的"字母异位词分组"问题很像。字母异位词就像是同一花色的扑克牌,它们由相同的字母组成,只是字母的排列顺序不同。

问题描述

LeetCode第49题"字母异位词分组"是这样描述的:给你一个字符串数组,请你将所有的字母异位词放在同一个组里。字母异位词是由相同的字母重新排列组成的单词。

例如,“eat”、"ate"和"tea"就是字母异位词,因为它们都由字母’a’、‘e’、't’组成,只是排列顺序不同。

最直观的解法:排序比较法

如何判断两个单词是否是字母异位词?最直观的方法就是:把两个单词的字母都排序后比较,如果排序后相同,那就是字母异位词。

就像整理扑克牌时,我们会把每张牌的花色符号位置摆正,这样就容易看出它们是不是同一花色。

具体步骤是这样的:

  1. 对于每个单词,将其字符排序得到一个标准形式
  2. 使用这个标准形式作为key,原单词作为value,存入哈希表
  3. 具有相同key的单词就是字母异位词,将它们分到同一组

让我们用一个例子来模拟这个过程:

  1. 输入:strs = ["eat","tea","tan","ate","nat","bat"]
  2. 处理"eat"
  3. - 排序后是"aet"
  4. - 哈希表:{"aet": ["eat"]}
  5. 处理"tea"
  6. - 排序后是"aet"
  7. - 找到已有的组,添加进去
  8. - 哈希表:{"aet": ["eat","tea"]}
  9. 处理"tan"
  10. - 排序后是"ant"
  11. - 创建新组
  12. - 哈希表:{"aet": ["eat","tea"], "ant": ["tan"]}
  13. ...以此类推

这种思路可以用Java代码这样实现:

  1. public List<List<String>> groupAnagrams(String[] strs) {
  2. // 创建哈希表,key是排序后的字符串,value是原字符串列表
  3. Map<String, List<String>> map = new HashMap<>();
  4. for (String str : strs) {
  5. // 将字符串转换为字符数组并排序
  6. char[] chars = str.toCharArray();
  7. Arrays.sort(chars);
  8. // 将排序后的字符数组转回字符串,作为key
  9. String key = new String(chars);
  10. // 如果key不存在,创建新的列表
  11. if (!map.containsKey(key)) {
  12. map.put(key, new ArrayList<>());
  13. }
  14. // 将原字符串添加到对应的组
  15. map.get(key).add(str);
  16. }
  17. // 返回所有分组的列表
  18. return new ArrayList<>(map.values());
  19. }

优化解法:计数法

还有一种更巧妙的解法。如果我们仔细观察,会发现:字母异位词的特点是每个字母出现的次数相同。

就像数纸牌时,我们不需要真的把牌排序,只需要数一下每种花色各有多少张。

计数法的原理

  1. 对于每个单词,统计其中每个字母出现的次数
  2. 将这个统计结果作为key(需要特殊的格式化方式)
  3. 相同统计结果的单词就是字母异位词

算法步骤(伪代码)

  1. 创建哈希表map存储分组结果
  2. 对于每个单词:
    • 创建一个大小为26的计数数组
    • 统计每个字母的出现次数
    • 将计数数组转换为特殊格式的字符串作为key
    • 将单词加入对应的分组
  3. 返回所有分组

示例运行

让我们模拟处理输入:[“eat”, “tea”, “tan”]

  1. 处理"eat"
  2. - 计数:{a:1, e:1, t:1, 其他:0}
  3. - key = "1#1#0#0#1#0#..."(代表a1b0c0d0e1...)
  4. - 创建新组:{"1#1#0#0#1#0...": ["eat"]}
  5. 处理"tea"
  6. - 计数相同:{a:1, e:1, t:1, 其他:0}
  7. - 找到相同的key,加入该组
  8. - {"1#1#0#0#1#0...": ["eat","tea"]}
  9. 处理"tan"
  10. - 计数:{a:1, n:1, t:1, 其他:0}
  11. - 新的key,创建新组
  12. - map添加新组

Java代码实现

  1. public List<List<String>> groupAnagrams(String[] strs) {
  2. // 创建哈希表存储分组
  3. Map<String, List<String>> map = new HashMap<>();
  4. for (String str : strs) {
  5. // 创建字符计数数组
  6. int[] count = new int[26];
  7. // 统计每个字符出现的次数
  8. for (char c : str.toCharArray()) {
  9. count[c - 'a']++;
  10. }
  11. // 构建key
  12. StringBuilder key = new StringBuilder();
  13. for (int i = 0; i < 26; i++) {
  14. key.append(count[i]).append('#');
  15. }
  16. String keyStr = key.toString();
  17. // 添加到对应的分组
  18. if (!map.containsKey(keyStr)) {
  19. map.put(keyStr, new ArrayList<>());
  20. }
  21. map.get(keyStr).add(str);
  22. }
  23. return new ArrayList<>(map.values());
  24. }

排序法vs计数法

让我们比较这两种解法:

排序法的时间复杂度是O(nklogk),其中n是单词数量,k是最长单词的长度。主要时间花在对每个单词进行排序上。它的优点是直观易懂,而且如果单词长度不大,性能也不错。

计数法的时间复杂度是O(nk),因为我们只需要遍历每个单词一次,统计字母出现次数。虽然实现稍微复杂一些,但在处理长单词时更有优势。

题目模式总结

这道题体现了一个重要的算法模式:通过某种标准形式来对对象分组

这种模式在实际编程中经常出现,比如:

  • 按文件大小分类存储文件
  • 对用户行为数据进行分组分析
  • 整理图书馆的图书分类

解决这类问题的通用思路是:

  1. 设计一个合适的标准形式(可以是排序后的结果,也可以是某种统计特征)
  2. 用哈希表存储分组结果
  3. 选择合适的数据结构来表示标准形式(字符串、数组等)

小结

通过这道题,我们不仅学会了如何解决字母异位词分组问题,更重要的是理解了"标准形式分组"这一重要的算法思想。这种思想在很多实际问题中都能派上用场。

记住,写代码时多思考如何把问题抽象成更一般的模式,这样才能举一反三,提高解决问题的能力!


作者:忍者算法
公众号:忍者算法

从整理扑克牌到字母异位词分组:一道巧妙的排序应用题 |LeetCode 49 字母异位词分组的更多相关文章

  1. LeetCode 49: 字母异位词分组 Group Anagrams

    LeetCode 49: 字母异位词分组 Group Anagrams 题目: 给定一个字符串数组,将字母异位词组合在一起.字母异位词指字母相同,但排列不同的字符串. Given an array o ...

  2. Java实现 LeetCode 49 字母异位词分组

    49. 字母异位词分组 给定一个字符串数组,将字母异位词组合在一起.字母异位词指字母相同,但排列不同的字符串. 示例: 输入: ["eat", "tea", & ...

  3. Leetcode 49.字母异位词分组

    字母异位词分组 给定一个字符串数组,将字母异位词组合在一起.字母异位词指字母相同,但排列不同的字符串. 示例: 输入: ["eat", "tea", " ...

  4. LeetCode 49. 字母异位词分组(Group Anagrams)

    题目描述 给定一个字符串数组,将字母异位词组合在一起.字母异位词指字母相同,但排列不同的字符串. 示例: 输入: ["eat", "tea", "ta ...

  5. LeetCode:字母异位词分组【16】

    LeetCode:字母异位词分组[16] 题目描述 给定一个字符串数组,将字母异位词组合在一起.字母异位词指字母相同,但排列不同的字符串. 示例: 输入: ["eat", &quo ...

  6. 【leetcode】字母异位词分组

    给定一个字符串数组,将字母异位词组合在一起.字母异位词指字母相同,但排列不同的字符串. 示例: 输入: ["eat", "tea", "tan&quo ...

  7. 【LeetCode】49. 字母异位词分组

    49. 字母异位词分组 知识点:字符串:哈希表 题目描述 给你一个字符串数组,请你将 字母异位词 组合在一起.可以按任意顺序返回结果列表. 字母异位词 是由重新排列源单词的字母得到的一个新单词,所有源 ...

  8. [LeetCode] 49. Group Anagrams 分组变位词

    Given an array of strings, group anagrams together. For example, given: ["eat", "tea& ...

  9. Java实现 LeetCode 748 最短完整词(字母拆分+暴力)

    748. 最短完整词 如果单词列表(words)中的一个单词包含牌照(licensePlate)中所有的字母,那么我们称之为完整词.在所有完整词中,最短的单词我们称之为最短完整词. 单词在匹配牌照中的 ...

  10. [LeetCode] 249. Group Shifted Strings 分组偏移字符串

    Given a string, we can "shift" each of its letter to its successive letter, for example: & ...

随机推荐

  1. 介绍 GGUF-my-LoRA

    随着 llama.cpp 对 LoRA 支持的重构,现在可以将任意 PEFT LoRA 适配器转换为 GGUF,并与 GGUF 基础模型一起加载运行. 为简化流程,我们新增了一个名为 GGUF-my- ...

  2. 1分钟学会如何提升PCIe通信速率,基于RK3568J + FPGA国产平台!

    测试数据汇总 表 1 PCIe总线介绍 PCIe,即PCI-Express(peripheral component interconnect express)是一种高速串行计算机扩展总线标准.主要用 ...

  3. CSS3 transform转换

    1.先说说css的坐标系: x轴的正方向就是水平向右的方向 y轴的正方向就是垂直向下的方向 z轴的正方向就是屏幕到用户的方向 2.位移 说明:位移是转换属性中的一个值,包含2d与3d 属性值 说明 t ...

  4. 在Python工具箱中,创建对应子工具集

    目录 问题描述 实现方法 问题描述 在Pro中,新建自定义工具箱后,直接通过操作可以添加工具集. 但是新建python工具箱后,却没有新建的操作.因为python工具箱的对象定义,都是在脚本中定义的, ...

  5. 鸿蒙UI系统组件01——文本组件(Text/Span)

    如果你也对鸿蒙开发感兴趣,加入"Harmony自习室"吧!点击下面的名片关注公众号. 1.概述 Text是文本组件,是我们开发UI界面中最常见的组件之一,通常用于展示用户的视图,如 ...

  6. Centos中keytool不起作用的解决方法

    ​keytool是Java开发中用于管理密钥和证书的工具,可以用于生成密钥.创建证书请求.导入和导出证书等操作.你可以在Oracle官网上下载和安装JDK,然后在JDK的 bin目录下找到 keyto ...

  7. 【WEB前端】【JQuery】搜索li标签的内容

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  8. maven:Could not transfer artifact from/to maven-default-http-blocker (http://0.0.0.0/): Blocked m...

    今天在拉完项目后拉取包的过程中,maven报错: Could not transfer artifact from/to 对应的包 maven-default-http-blocker (http:/ ...

  9. okhttp3设置代理(http/https)

    最近项目网络请求需要设置代理,记录一下.http和https都可以. OkHttpClient.Builder builder = new OkHttpClient.Builder(); //代理服务 ...

  10. Redis-十大数据类型

    Reids数据类型指的是value的类型,key都是字符串 redis-server:启动redis服务 redis-cli:进入redis交互式终端 常用的key的操作 redis的命令和参数不区分 ...