当前位置: 首页 > news >正文

【LeetCode 热题 100】49. 字母异位词分组

Problem: 49. 字母异位词分组

文章目录

  • 整体思路
  • 完整代码
  • 时空复杂度
        • 时间复杂度:O(N * K log K)
    • 空间复杂度:O(N * K)

整体思路

这段代码旨在解决一个经典的字符串问题:字母异位词分组 (Group Anagrams)。问题要求将一个字符串数组中的所有字符串,按照它们是否互为字母异位词(Anagrams)来进行分组。字母异位词是指由相同的字符以不同的顺序构成的字符串(例如,"eat""tea")。

该算法采用了一种非常经典且高效的策略:使用排序后的字符串作为“规范哈希键”。其核心逻辑步骤如下:

  1. 核心思想:异位词的“身份证”

    • 算法的基石在于一个关键的观察:所有互为字母异位词的字符串,当它们内部的字符被排序后,都会得到完全相同的字符串。
    • 例如,"eat", "tea", "ate" 这三个异位词,排序后都变成了 "aet"
    • 因此,这个排序后的字符串 "aet" 就可以作为这一组异位词的唯一标识,或者说“身份证”。
  2. 数据结构选择

    • 算法选择了一个 哈希表 (HashMap)Map<String, List<String>>,来存储分组结果。
    • 键 (Key):哈希表的键是排序后的“规范字符串”(如 "aet")。
    • 值 (Value):哈希表的值是一个字符串列表,用来存放所有共享同一个“规范字符串”的原始字符串(如 ["eat", "tea", "ate"])。
  3. 算法步骤

    • 遍历输入数组:算法遍历输入的字符串数组 strs 中的每一个字符串 s
    • 生成哈希键:对于每个字符串 s
      a. 将其转换为字符数组 char[] sortedS = s.toCharArray();
      b. 对这个字符数组进行排序 Arrays.sort(sortedS);
      c. 将排序后的字符数组转换回字符串 new String(sortedS),这就是该字符串的“规范哈希键”。
    • 分组
      a. 使用这个“规范哈希键”去哈希表中查找。
      b. m.computeIfAbsent(key, _ -> new ArrayList<>()) 是一个非常简洁的Java 8+的写法。它的作用是:如果 key 不存在,就创建一个新的 ArrayList 并与 key 关联;如果 key 已存在,就直接返回与 key 关联的现有 ArrayList
      c. .add(s):无论上面是新建还是获取,最终都会将原始字符串 s 添加到对应的列表中。
    • 返回结果:遍历结束后,哈希表 mvalues() 集合中就包含了所有分组好的列表。将 m.values() 转换为一个新的 ArrayList 并返回。

完整代码

import java.util.*;class Solution {/*** 将字符串数组按照字母异位词进行分组。* @param strs 输入的字符串数组* @return 分组后的字符串列表*/public List<List<String>> groupAnagrams(String[] strs) {// 创建一个哈希表,键是排序后的规范字符串,值是属于该组的原始字符串列表。Map<String, List<String>> m = new HashMap<>();// 遍历输入的每一个字符串for (String s : strs) {// 步骤 1: 生成“规范哈希键”// 将当前字符串转换为字符数组char[] sortedS = s.toCharArray();// 对字符数组进行排序Arrays.sort(sortedS);// 将排序后的字符数组转换回字符串,作为哈希表的键String key = new String(sortedS);// 步骤 2: 将原始字符串放入对应的分组// computeIfAbsent(key, k -> new ArrayList<>()) 的语义:// - 如果 map 中不存在 key,则执行 lambda 表达式创建一个 new ArrayList<>(),//   并将其作为 value 与 key 关联,然后返回这个新创建的 list。// - 如果 map 中已存在 key,则直接返回与 key 关联的现有 list。// 接着,.add(s) 将原始字符串 s 加入到获取到的 list 中。m.computeIfAbsent(key, _ -> new ArrayList<>()).add(s);}// 步骤 3: 返回结果// m.values() 返回哈希表中所有值的集合(即所有分组的 List<String>)。// 将这个集合转换为一个新的 ArrayList 并返回。return new ArrayList<>(m.values());}
}

时空复杂度

时间复杂度:O(N * K log K)
  1. 变量定义
    • N:输入数组 strs 中字符串的数量。
    • K:数组中字符串的最大长度。
  2. 外层循环for (String s : strs) 循环执行 N 次。
  3. 循环内部操作:这是分析的关键。
    • s.toCharArray():将长度为 k 的字符串转为字符数组,时间复杂度为 O(k)
    • Arrays.sort(sortedS):对一个长度为 k 的字符数组进行排序,时间复杂度为 O(k log k)
    • new String(sortedS):将长度为 k 的字符数组转为字符串,时间复杂度为 O(k)
    • m.computeIfAbsent(...)add(...):哈希表的插入和获取操作,平均时间复杂度是 O(k),因为哈希键(字符串)的比较和哈希码计算的成本与字符串长度 k 成正比。

综合分析
每次循环的成本由其中最耗时的操作决定,即排序操作 O(k log k)
由于这个操作在外层循环中重复了 N 次,所以总的时间复杂度是 O(N * K log K)

空间复杂度:O(N * K)

  1. 主要存储开销:哈希表 m 是主要的额外空间消耗。
  2. 空间大小
    • 在最坏的情况下,如果所有字符串都互不为异位词,那么哈希表中将有 N 个条目。
    • 每个条目的是一个排序后的字符串,其长度最多为 K
    • 每个条目的是一个列表,其中包含原始字符串,其长度也最多为 K
    • 因此,哈希表需要存储所有 N 个字符串(以原始形式或排序形式)。所有字符串的总长度可以表示为 N * K_avgK_avg为平均长度),在最坏情况下为 O(N * K)
  3. 临时变量:在循环中,char[] sortedS 需要 O(K) 的临时空间。但这部分空间在每次循环后都会被回收,并且其最大值 O(K) 不会超过哈希表的总空间 O(N*K)。

综合分析
算法所需的额外空间主要由哈希表决定,它存储了所有原始字符串的信息。因此,空间复杂度为 O(N * K)

参考灵神

http://www.xdnf.cn/news/1477981.html

相关文章:

  • Windows 11 手动下载安装配置 uv、配置国内源
  • 固定资产管理系统(vue+Springboot+mybatis)
  • 行为式验证码技术解析:滑块拼图、语序选词与智能无感知
  • Vllm-0.10.1:vllm bench serve参数说明
  • 【完整源码+数据集+部署教程】农作物病害检测系统源码和数据集:改进yolo11-HSFPN
  • Flutter常用库集锦
  • Webpack热更新(HMR)底层原理详解
  • 基于定制开发开源AI智能名片S2B2C商城小程序的DMP平台离线文件上传功能优化研究
  • RK3568 Trust
  • 进程间通信(IPC)方式
  • AgentScope 1.0深度解析:技术架构、使用教程与多智能体开发实践
  • 跟着开题报告学答辩!《 Access学情分析系统的设计与实现》开题答辩实录分享!
  • Linux系统编程守护进程(36)
  • Linux笔记---TCP套接字编程
  • Docker学习笔记-网络类型
  • 【干货推荐】AI助理前端UI组件-悬浮球组件
  • 下载数据集用于图像分类并自动分为训练集和测试集方法
  • Python零基础速成指南:12周从小白到项目实战
  • uniapp | 解决组件样式不生效问题
  • uniapp新增页面及跳转配置方法
  • 【最新版】超级好用的软件卸载工具IObit Uninstaller v15.0.0.8 中文解压即用版 告别残留烦恼
  • 力扣p2009 使数组连续的最少操作数 详解
  • ELFK:企业级日志管理的完整解决方案——从入门到精通
  • 尚硅谷宋红康JVM全套教程(详解java虚拟机)
  • 苍穹外卖项目实战(day-5完整版)-记录实战教程及问题的解决方法
  • 2025高教社国赛数学建模C题参考论文(含模型和代码)
  • 【面试向】人工智能机器学习介绍
  • 【51单片机-B030】【protues仿真】基于51单片机万年历系统
  • 心路历程-passwdusermod命令补充
  • 嵌入式学习——ARM 体系架构1