Python面试宝典第33题:电话号码的字母组合

题目

给定一个仅包含数字2-9的字符串,返回所有它能表示的字母组合,答案可以按任意顺序返回。给出数字到字母的映射如下图(与手机按键相同)。注意:1不对应任何字母。

示例 1:

python 复制代码
输入:digits = "23"
输出:["ad","ae","af","bd","be","bf","cd","ce","cf"]

示例 2:

python 复制代码
输入:digits = ""
输出:[]

示例 3:

python 复制代码
输入:digits = "2"
输出:["a","b","c"]

递归法

使用递归法求解本题的基本思想是:建立数字到字母的映射关系(比如:数字2对应字母"abc");对于给定的数字串,从左到右逐个数字考虑,针对每个数字生成对应的字母,并与之前已有的组合进行连接形成新的组合。使用递归法求解本题的主要步骤如下。

1、创建一个字典phone,用于存储数字到字母的映射。

2、定义一个递归函数backtrack,它接受当前组合combination和剩余未处理的数字串next_digits作为参数。

3、如果next_digits为空,说明当前组合已完成,将其加入结果列表。

4、否则,取出next_digits的第一个数字,并获取该数字对应的字母,然后对这些字母进行循环,将每个字母添加到combination的末尾,并递归地处理剩余的数字。

5、在递归返回后,从combination中移除最后添加的字母,以便于下一次迭代可以尝试其他字母。

根据上面的算法步骤,我们可以得出下面的示例代码。

python 复制代码
def letter_combinations_by_recursion(digits):
    # 定义数字到字母的映射
    phone = {'2': 'abc', '3': 'def', '4': 'ghi', '5': 'jkl', 
             '6': 'mno', '7': 'pqrs', '8': 'tuv', '9': 'wxyz'}
    result = []

    # 递归函数
    def backtrack(combination, next_digits):
        # 如果没有剩余的数字
        if len(next_digits) == 0:
            result.append(combination)
        else:
            # 获取下一个数字对应的字母
            for letter in phone[next_digits[0]]:
                # 递归调用
                backtrack(combination + letter, next_digits[1:])

    if not digits:
        return []

    backtrack("", digits)
    return result
    
print(letter_combinations_by_recursion("23"))
print(letter_combinations_by_recursion(""))
print(letter_combinations_by_recursion("2"))

迭代法

使用迭代法求解本题的基本思想是:利用栈(stack)或队列(queue)来模拟递归过程中的状态转移。解题的主要步骤如下。

1、初始化。创建一个队列queue,并将初始的空字符串加入队列。同时,创建一个字典phone,用于存储数字到字母的映射。

2、迭代处理。从输入的数字串的第一个数字开始,每次取出队列中的元素,并将当前数字对应的字母逐一添加到这些元素后面,形成新的组合,最后将这些新组合加入队列。

3、终止条件。当所有数字都已经被处理过,队列中的元素即为最终的结果。

根据上面的算法步骤,我们可以得出下面的示例代码。

python 复制代码
from collections import deque

def letter_combinations_by_iteration(digits):
    # 定义数字到字母的映射
    phone = {'2': 'abc', '3': 'def', '4': 'ghi', '5': 'jkl', 
             '6': 'mno', '7': 'pqrs', '8': 'tuv', '9': 'wxyz'}
    result = []

    # 如果输入的字符串为空,直接返回空列表
    if not digits:
        return result

    # 创建一个队列,并将初始的空字符串加入队列
    queue = deque([''])

    # 遍历输入的数字串
    for digit in digits:
        # 队列的长度会随着迭代而变化,这里记录当前长度
        size = len(queue)
        # 处理当前数字对应的字母
        for _ in range(size):
            # 取出队列中的组合
            combination = queue.popleft()
            # 将当前数字对应的字母逐一添加到组合后面
            for letter in phone[digit]:
                # 形成新的组合并加入队列
                queue.append(combination + letter)

    while queue:
        result.append(queue.popleft())

    return result

print(letter_combinations_by_iteration("23"))
print(letter_combinations_by_iteration(""))
print(letter_combinations_by_iteration("2"))

总结

递归法求解本题时,每个数字对应最多4个字母,所以递归的深度是输入数字的长度,每层递归需要遍历该数字对应的字母数量。因此,时间复杂度是 O(4^n)。其中,n是输入数字的长度。递归调用栈的最大深度与输入数字的长度相同,因此空间复杂度为O(n)。

与递归法一样,迭代法的时间复杂度也是O(4^n)。迭代法使用队列存储了中间结果,空间复杂度为O(n * 4^n)。这是因为,在最坏情况下,队列可能需要存储所有可能的组合。

相关推荐
Envyᥫᩣ7 分钟前
Python中的自然语言处理:从基础到高级
python·自然语言处理·easyui
Amor风信子7 分钟前
华为OD机试真题---跳房子II
java·数据结构·算法
哪 吒7 分钟前
华为OD机试 - 几何平均值最大子数(Python/JS/C/C++ 2024 E卷 200分)
javascript·python·华为od
我是陈泽10 分钟前
一行 Python 代码能实现什么丧心病狂的功能?圣诞树源代码
开发语言·python·程序员·编程·python教程·python学习·python教学
hakesashou11 分钟前
python全栈开发是什么?
python
戊子仲秋24 分钟前
【LeetCode】每日一题 2024_10_2 准时到达的列车最小时速(二分答案)
算法·leetcode·职场和发展
邓校长的编程课堂26 分钟前
助力信息学奥赛-VisuAlgo:提升编程与算法学习的可视化工具
学习·算法
创作小达人30 分钟前
家政服务|基于springBoot的家政服务平台设计与实现(附项目源码+论文+数据库)
开发语言·python
ZPC821040 分钟前
Python使用matplotlib绘制图形大全(曲线图、条形图、饼图等)
开发语言·python·matplotlib
镜花照无眠42 分钟前
Python爬虫使用实例-mdrama
开发语言·爬虫·python