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

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. 时间 / 空间复杂度最优,符合题目要求。

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

http://www.cnnetsun.cn/news/42731.html

相关文章:

  • 告别开题报告模板拼凑!虎贲等考 AI 智能生成,让选题逻辑从模糊想法变身可执行研究计划
  • 【LeetCode刷题】跳跃游戏
  • 鸿蒙PC UI控件库 - PasswordInput 密码输入框详解
  • day37简单的神经网络@浙大疏锦行
  • 【水果识别】基于机器视觉苹果和香蕉的成熟度和大小检测附Matlab代码
  • JAVA的平凡之路——此峰乃是最高峰JVM-附加小菜-04
  • 【电力系统】电力系统优化与控制热液调度附Matlab代码和报告
  • 基于6种最新算法(小龙虾优化算法COA、MSA、RTH、NOA、BFO、SWO)求解机器人路径规划研究附Matlab代码
  • Golang实战:构建综合多头(逾期+反欺诈)风险查询的高性能客户端
  • 【TSP问题】基于蜣螂算法DBO和改进的蜣螂算法FADBO求解旅行商TSP问题(可根据自己的经纬度设置自己想要到达的地区)附Matlab代码
  • 【太阳能学报EI复现】基于粒子群优化算法的风-水电联合优化运行分析附Matlab代码
  • 数据结构:二叉排序树,平衡二叉树,红黑树的介绍
  • 软件复用的分类与实现
  • google服务
  • 进程PCB
  • 实战教程:1小时掌握逆向Unity游戏 (共13课时)
  • [从零构建操作系统]08 函数调用时栈的底层行为解析
  • 力扣hot100:搜索插入位置
  • Java冷启动全指南:从原理到实战优化
  • 测试 - 单元测试(JUnit)
  • C++中多态
  • c++经典练习题-多分支
  • qt为什么转向用cmake放弃qmake
  • 云屋音视频 SDK 凭何成为信创技术困局的 “破局者”?
  • 纯电动汽车动力经济性仿真:Cruise与Simulink联合仿真(2015版),包含BMS、再...
  • 【怎么理解maven中的镜像和仓库?】
  • comsol枝晶生长,沉积模型,包括:典型,形状成核,随机成核,均匀沉积,雪花晶形成过程。 适...
  • 终极指南:Qwen3-30B-A3B多GPU分布式推理完整解决方案
  • 腾讯混元语音驱动数字人技术:重塑动态视频生成新范式
  • 【MicroPython编程-ESP32篇】-Web页面显示DHT11传感器数据