news 2026/4/23 14:43:09

LeetCode 17. 电话号码的字母组合 | 深度解析 + 高效回溯实现

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
LeetCode 17. 电话号码的字母组合 | 深度解析 + 高效回溯实现

一、题目介绍

1.1 题目描述

给定一个仅包含数字2-9的字符串,返回所有它能表示的字母组合。答案可以按任意顺序返回。

数字到字母的映射与电话按键一致(1 不对应任何字母):

  • 2: abc
  • 3: def
  • 4: ghi
  • 5: jkl
  • 6: mno
  • 7: pqrs
  • 8: tuv
  • 9: wxyz

1.2 示例

  • 示例 1:输入:digits = "23"输出:["ad","ae","af","bd","be","bf","cd","ce","cf"]

  • 示例 2:输入:digits = "2"输出:["a","b","c"]

  • 示例 3:输入:digits = ""输出:[]

1.3 题目难度

中等

1.4 核心考点

  • 回溯算法(深度优先搜索 DFS)的应用
  • 字符串的遍历与组合构建
  • 递归思想的落地实现

二、解题思路

2.1 核心思想

这是一道典型的组合型回溯问题,核心逻辑是:

  1. 为每个数字建立固定的字母映射表;
  2. 递归遍历每个数字对应的字母,逐步构建字母组合(路径);
  3. 当路径长度等于输入数字的长度时,说明已构建完成一个完整组合,将其加入结果集;
  4. 回溯过程中复用同一个路径字符串,通过覆盖的方式减少内存开销。

2.2 算法流程

  1. 处理边界条件:若输入字符串为空,直接返回空数组;
  2. 初始化结果数组和路径字符串(路径长度与输入数字长度一致);
  3. 递归函数(DFS):
    • 终止条件:当前处理到第n个数字(所有数字处理完毕),将路径加入结果;
    • 遍历当前数字对应的所有字母,将字母填入路径的对应位置,递归处理下一个数字;
  4. 最终返回结果数组。

三、完整代码实现

#include <iostream> #include <vector> #include <string> using namespace std; class Solution { // 数字到字母的映射表,索引对应数字0-9 static constexpr string MAPPING[10] = {"", "", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"}; public: vector<string> letterCombinations(string digits) { int n = digits.length(); // 边界条件:空字符串直接返回空数组 if (n == 0) { return {}; } vector<string> ans; // 存储最终结果 string path(n, 0); // 路径字符串,长度固定为n,避免频繁拼接 // 递归lambda表达式(C++14及以上支持this auto&&) auto dfs = [&](this auto&& dfs, int i) -> void { // 递归终止:处理完所有数字 if (i == n) { ans.emplace_back(path); // 将当前路径加入结果 return; } // 获取当前数字对应的字母串 const string& letters = MAPPING[digits[i] - '0']; // 遍历当前数字的所有字母 for (char c : letters) { path[i] = c; // 直接覆盖路径的第i位,无需拼接/回退 dfs(i + 1); // 递归处理下一个数字 } }; dfs(0); // 从第0个数字开始递归 return ans; } }; // 测试函数 void test() { Solution s; // 测试案例1 vector<string> res1 = s.letterCombinations("23"); cout << "输入: 23 | 输出: "; for (const string& str : res1) { cout << str << " "; } cout << endl; // 测试案例2 vector<string> res2 = s.letterCombinations("2"); cout << "输入: 2 | 输出: "; for (const string& str : res2) { cout << str << " "; } cout << endl; // 测试案例3 vector<string> res3 = s.letterCombinations(""); cout << "输入: 空 | 输出: " << (res3.empty() ? "空数组" : "非空") << endl; } int main() { test(); return 0; }

四、代码深度解析

4.1 映射表定义

static constexpr string MAPPING[10] = {"", "", "abc", "def", "ghi", "jkl", "mno", "pqrs", "tuv", "wxyz"};
  • 使用static constexpr定义常量映射表,避免每次调用函数时重复创建,提升性能;
  • 索引 0 和 1 对应空字符串(因为 0、1 无字母映射),索引 2-9 对应电话按键的字母。

4.2 边界条件处理

if (n == 0) { return {}; }
  • 输入为空字符串时,直接返回空数组,符合题目要求。

4.3 路径字符串优化

string path(n, 0);
  • 传统回溯通常使用空字符串逐步拼接,再在递归返回时「回退」(pop_back);
  • 此处直接初始化长度为n的路径字符串,通过覆盖指定位置的方式构建组合,无需回退操作,减少字符串拼接 / 删除的开销。

4.4 递归 Lambda 表达式

auto dfs = [&](this auto&& dfs, int i) -> void { ... };
  • C++14 引入的「泛型 lambda」+「this 捕获」,实现递归 lambda(无需额外定义全局 / 类内递归函数);
  • this auto&& dfs表示 lambda 自身的引用,用于递归调用;
  • 捕获列表[&]表示按引用捕获外部变量(ans、path、digits、MAPPING)。

4.5 递归核心逻辑

if (i == n) { ans.emplace_back(path); return; } for (char c : letters) { path[i] = c; dfs(i + 1); }
  • 终止条件:i == n表示所有数字处理完毕,将当前路径加入结果集;
  • 遍历当前数字的所有字母,将字母填入路径的第i位,递归处理下一个数字(i+1);
  • 由于路径是覆盖式修改,无需手动「回溯」(传统回溯的pop_back操作),代码更简洁。

五、测试结果

输入: 23 | 输出: ad ae af bd be bf cd ce cf 输入: 2 | 输出: a b c 输入: 空 | 输出: 空数组

完全符合题目预期。

六、算法复杂度分析

6.1 时间复杂度

  • 假设输入数字的长度为n,每个数字对应k个字母(k∈[3,4]);
  • 总组合数为3^n4^n(取决于数字对应的字母数);
  • 时间复杂度:O(3^n ~ 4^n),即所有组合的总数。

6.2 空间复杂度

  • 递归调用栈的深度为n(数字长度);
  • 路径字符串占用O(n)空间;
  • 结果数组占用O(3^n ~ 4^n)空间(存储所有组合);
  • 总空间复杂度:O(3^n ~ 4^n)

七、扩展与优化

7.1 非递归(迭代)实现

如果不想使用递归,也可以通过迭代的方式实现:

vector<string> letterCombinations_iter(string digits) { if (digits.empty()) return {}; vector<string> ans = {""}; for (char d : digits) { vector<string> temp; const string& letters = MAPPING[d - '0']; for (const string& s : ans) { for (char c : letters) { temp.push_back(s + c); } } ans.swap(temp); } return ans; }

7.2 优化点说明

  1. 本文的递归实现使用「固定长度路径 + 覆盖式修改」,比传统的「拼接 + 回退」减少了字符串操作的开销;
  2. 映射表使用static constexpr,避免重复初始化,提升性能;
  3. 递归 lambda 比单独定义递归函数更简洁,代码内聚性更强。

八、总结

本题是回溯算法的经典入门题,核心在于理解「逐步构建组合 + 递归遍历」的思想。本文提供的实现方案有以下特点:

  1. 代码简洁高效,利用 C++14 的 lambda 递归简化代码结构;
  2. 路径字符串优化,避免频繁拼接和回退操作;
  3. 边界条件处理完善,覆盖空输入等特殊情况;
  4. 时间 / 空间复杂度最优,符合题目要求。

掌握本题的回溯思想后,可进一步解决类似的组合问题(如组合总和、子集等),是算法学习中不可或缺的基础知识点。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 10:10:02

AI 写论文哪家强?实测 5 款主流工具后我们发现:真正适配毕业论文的,从来不是 “写得快”,而是 “写得稳、查得到、改得了”

&#x1f6a8; 开篇&#xff1a;论文 AI 工具的 “速度陷阱”&#xff01;90% 的学生选错了核心标准​ “10 分钟生成 8000 字初稿”“3 天搞定毕业论文”—— 这些看似诱人的宣传&#xff0c;让不少毕业生陷入 “唯速度论” 的选择误区。但实测发现&#xff0c;那些主打 “写…

作者头像 李华
网站建设 2026/4/23 10:09:38

【自然语言处理】自然语言理解:从技术基础到多元应用的全景探索

目录 一、引言 二、自然语言理解的基础&#xff1a;语言的复杂性与技术挑战 1. 语言结构的差异&#xff1a;从分词到词形变化 2. 低资源语言的生存困境 3. 字符编码的兼容性问题 三、对话式人工智能&#xff1a;人机交互的核心模块 1. 从语音到文本&#xff1a;语音识别&…

作者头像 李华
网站建设 2026/4/23 10:09:38

wpf 类图

WPF的实现架构比较抽象&#xff0c;我先放一放。 WPF的命名空间都是System.Window开头。还有一张wpf类图其中比较核心的类是FrameworkElement&#xff0c;它派生自UIElement&#xff0c;具有&#xff1a;数据绑定、样式、资源等wpf最重要的功能。目前我最熟悉的类是Panel和Cont…

作者头像 李华
网站建设 2026/4/23 9:03:48

【Agent】Evaluation and Benchmarking of LLM Agents: A Survey

note 文章目录note一、论文想解决什么问题&#xff1f;&#xff08;Why&#xff09;核心问题二、论文的核心贡献&#xff08;What&#xff09;1️⃣ 提出一个 **二维评测分类体系&#xff08;Taxonomy&#xff09;**2️⃣ 系统梳理已有工作3️⃣ 明确指出 **企业级 Agent 评测的…

作者头像 李华