📑 题目:127. 单词接龙

🚀 本题 LeetCode 传送门

题目大意

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

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

说明:

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

解题思路

  • 这一题要求输出从 beginWord 变换到 endWord 最短变换次数。可以用 BFS,从 beginWord 开始变换,把该单词的每个字母都用 'a'~'z' 变换一次,生成的数组到 wordList 中查找,这里用 Map 来记录查找。找得到就入队列,找不到就输出 0 。入队以后按照 BFS 的算法依次遍历完,当所有单词都 len(queue)<=0 出队以后,整个程序结束。

  • 这一题题目中虽然说了要求找到一条最短的路径,但是实际上最短的路径的寻找方法已经告诉你了:

    1. 每次只变换一个字母
    2. 每次变换都必须在 wordList
      所以不需要单独考虑何种方式是最短的。

代码

  1. package leetcode
  2. func ladderLength(beginWord string, endWord string, wordList []string) int {
  3. wordMap, que, depth := getWordMap(wordList, beginWord), []string{beginWord}, 0
  4. for len(que) > 0 {
  5. depth++
  6. qlen := len(que)
  7. for i := 0; i < qlen; i++ {
  8. word := que[0]
  9. que = que[1:]
  10. candidates := getCandidates(word)
  11. for _, candidate := range candidates {
  12. if _, ok := wordMap[candidate]; ok {
  13. if candidate == endWord {
  14. return depth + 1
  15. }
  16. delete(wordMap, candidate)
  17. que = append(que, candidate)
  18. }
  19. }
  20. }
  21. }
  22. return 0
  23. }
  24. func getWordMap(wordList []string, beginWord string) map[string]int {
  25. wordMap := make(map[string]int)
  26. for i, word := range wordList {
  27. if _, ok := wordMap[word]; !ok {
  28. if word != beginWord {
  29. wordMap[word] = i
  30. }
  31. }
  32. }
  33. return wordMap
  34. }
  35. func getCandidates(word string) []string {
  36. var res []string
  37. for i := 0; i < 26; i++ {
  38. for j := 0; j < len(word); j++ {
  39. if word[j] != byte(int('a')+i) {
  40. res = append(res, word[:j]+string(int('a')+i)+word[j+1:])
  41. }
  42. }
  43. }
  44. return res
  45. }