leetcode 127. 单词接龙

127. 单词接龙
给定两个单词(beginWord 和 endWord)和一个字典,找到从 beginWord 到 endWord 的最短转换序列的长度。转换需遵循如下规则:

每次转换只能改变一个字母。
转换过程中的中间单词必须是字典中的单词。
说明:

如果不存在这样的转换序列,返回 0。
所有单词具有相同的长度。
所有单词只由小写字母组成。
字典中不存在重复的单词。
你可以假设 beginWord 和 endWord 是非空的,且二者不相同。
示例 1:

输入:
beginWord = “hit”,
endWord = “cog”,
wordList = [“hot”,“dot”,“dog”,“lot”,“log”,“cog”]

输出: 5

解释: 一个最短转换序列是 “hit” -> “hot” -> “dot” -> “dog” -> “cog”,
返回它的长度 5。
示例 2:

输入:
beginWord = “hit”
endWord = “cog”
wordList = [“hot”,“dot”,“dog”,“lot”,“log”]

输出: 0

解释: endWord “cog” 不在字典中,所以无法进行转换。


其实就是将beginWord 单词作为树的根节点,依次向下遍历,看这棵树的叶子结点有没有endWord (有时候也不一定是叶子结点)。如果存在,返回其深度depth,反之返回0。此处有一点需要注意:

在树上层出现过的字符串没必要在下层再次出现,因为如果该字符串是转换过程中必须经过的中间字符串,那么应该挑选上层的该字符串继续进行变化,它的转换次数少。

因为题目所求的是最短转换序列的长度,所以一定要记住这一点。

另外还有一点就是这个转换规则:每次转换只能改变一个字母。每一次迭代中如何来找当前单词的转换单词呢?这里面所用的方法有很多种,我看了网上的一些帖子,大致分为两种。一种是将改变的字母按照小写字母排列情况分为26种情况,依次填进去进行判断。另一种是将改变字母的那个位置用“_”代替,比如“hit”要改变第二个位置的字母,则可表示为“h_t”,而“hot”改变第二个位置的字母后也可表示为“h_t”,说明这两个单词是可以直接转换的。

第一次尝试做一下,结果倒是正确,就是最后提交的时候结果超出了时间限制,那个测试例子是真的长 ∩=∩ ,还能怎么办,改呗。

class Solution:
    def ladderLength(self, beginWord: str, endWord: str, wordList: List[str]) -> int:
        num = len(beginWord)
        queue = collections.deque()
        queue.append(beginWord)
        used = []


        def is_valid(word, pattern):#判断待加入的单词是否符合加入条件:两者不等,只有一个字符不同
            count = 0
            if word == pattern and word != endWord:
                return False
            for i in range(num):
                if word[i] == pattern[i]:
                    count += 1
            if count >= num - 1:
                return True
            else:
                return False
            
        def solve( beginWord, endWord, wordList):#BFS,每一将符合条件的字符加入队列
            cnt = 1
            while queue:
                for _ in range(len(queue)):#本次队列有多少数,就循环多少次
                    node = queue.popleft()
                    used.append(node)#设置一个使用过的列表记录
                    
                    for item in wordList:
                        if is_valid(node, item) and item not in used and item not in queue:
                            if item == endWord:
                                return cnt+1
                            queue.append(item)
                            # print(queue)
                cnt += 1
            return 0

        return solve(beginWord, endWord, wordList)

后来看了下题解,主要有两种方法解决超时问题

  1. 我们在单词匹配是否只差一个字母的时候用了太多的计算时间, dog, cog,只有中间一位不同,我们可以建立一个字典,其中 *og可以表示后两位位og的三位单词,这样做的话,以来就建立了一个字典,如果右只差一位的单词,就会被聚合到一个字典的键值下面。在进行BFS查找匹配的时候,我们的效率会加快很多
    在这里插入图片描述
    在这里插入图片描述

  2. 还有一种方法就是双向BFS,不止是单向从 begin 到 end, 我们也从 end 查找 begin,这个方法但是我现在还没仔细看,等下次回过头再看。

class Solution:
    def ladderLength(self, beginWord: str, endWord: str, wordList: List[str]) -> int:
        # BFS
        from collections import deque, defaultdict
        # 先验判断
        if endWord not in wordList:
            return 0
        # 提前构建邻接表 -> 用generic state做key
        intermidiateWords = defaultdict(list)
        wordLen = len(beginWord)
        for word in wordList:
            for i in range(wordLen):
                intermidiateWords[word[:i] + '*' + word[i+1:]].append(word)
        # 建队列 加初始状态
        queue = deque()
        memo = set()
        queue.append(beginWord)
        memo.add(beginWord)
        step = 1
        while queue:
            size = len(queue)
            for _ in range(size):
                curWord = queue.popleft()
                for i in range(wordLen):
                    intermidiateCurWord = curWord[:i] + '*' + curWord[i+1:]
                    # 下一个状态的所有word
                    for word in intermidiateWords[intermidiateCurWord]:
                        if word == endWord:
                            return step + 1
                        if word not in memo:
                            queue.append(word)
                            memo.add(word)
            step += 1
        else:
            return 0

作者:cui-jin-hao-_official
链接:https://leetcode-cn.com/problems/word-ladder/solution/python-bfs-shuang-xiang-bfs-by-cui-jin-hao-_offici/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值