LeetCode - Word Ladder II

3 minute read

Problem description

description

Given two words (beginWord and endWord), and a dictionary’s word list, find all shortest transformation sequence(s) from beginWord to endWord, such that:

Only one letter can be changed at a time Each transformed word must exist in the word list. Note that beginWord is not a transformed word. Note:

Return an empty list if there is no such transformation sequence. All words have the same length. All words contain only lowercase alphabetic characters. You may assume no duplicates in the word list. You may assume beginWord and endWord are non-empty and are not the same. Example 1:

1
2
3
4
5
6
7
8
9
10
Input:
beginWord = "hit",
endWord = "cog",
wordList = ["hot","dot","dog","lot","log","cog"]

Output:
[
  ["hit","hot","dot","dog","cog"],
  ["hit","hot","lot","log","cog"]
]

Example 2:

1
2
3
4
5
6
7
8
Input:
beginWord = "hit"
endWord = "cog"
wordList = ["hot","dot","dog","lot","log"]

Output: []

Explanation: The endWord "cog" is not in wordList, therefore no possible transformation.

Analysis

The idea is to use bfs to find the shortest path. And use dfs with distance to find the final solution.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Queue;
import java.util.Set;

public class WordLadderii {
  class Solution {
    public List<List<String>> findLadders(String start, String end, List<String> wordList) {
      HashSet<String> dict = new HashSet<String>(wordList);
      List<List<String>> res = new ArrayList<List<String>>();
      HashMap<String, ArrayList<String>> nodeNeighbors =
          new HashMap<String, ArrayList<String>>(); // Neighbors for every node
      HashMap<String, Integer> distance =
          new HashMap<String, Integer>(); // Distance of every node from the start node
      ArrayList<String> solution = new ArrayList<String>();

      dict.add(start);
      bfs(start, end, dict, nodeNeighbors, distance);
      dfs(start, end, dict, nodeNeighbors, distance, solution, res);
      return res;
    }

    // BFS: Trace every node's distance from the start node (level by level).
    private void bfs(
        String start,
        String end,
        Set<String> dict,
        HashMap<String, ArrayList<String>> nodeNeighbors,
        HashMap<String, Integer> distance) {
      for (String str : dict) nodeNeighbors.put(str, new ArrayList<String>());

      Queue<String> queue = new LinkedList<String>();
      queue.offer(start);
      distance.put(start, 0);

      while (!queue.isEmpty()) {
        int count = queue.size();
        boolean foundEnd = false;
        for (int i = 0; i < count; i++) {
          String cur = queue.poll();
          int curDistance = distance.get(cur);
          ArrayList<String> neighbors = getNeighbors(cur, dict);

          for (String neighbor : neighbors) {
            nodeNeighbors.get(cur).add(neighbor);
            if (!distance.containsKey(neighbor)) { // Check if visited
              distance.put(neighbor, curDistance + 1);
              if (end.equals(neighbor)) // Found the shortest path
              foundEnd = true;
              else queue.offer(neighbor);
            }
          }
        }

        if (foundEnd) break;
      }
    }

    // Find all next level nodes.
    private ArrayList<String> getNeighbors(String node, Set<String> dict) {
      ArrayList<String> res = new ArrayList<String>();
      char chs[] = node.toCharArray();

      for (char ch = 'a'; ch <= 'z'; ch++) {
        for (int i = 0; i < chs.length; i++) {
          if (chs[i] == ch) continue;
          char old_ch = chs[i];
          chs[i] = ch;
          if (dict.contains(String.valueOf(chs))) {
            res.add(String.valueOf(chs));
          }
          chs[i] = old_ch;
        }
      }
      return res;
    }

    // DFS: output all paths with the shortest distance.
    private void dfs(
        String cur,
        String end,
        Set<String> dict,
        HashMap<String, ArrayList<String>> nodeNeighbors,
        HashMap<String, Integer> distance,
        ArrayList<String> solution,
        List<List<String>> res) {
      solution.add(cur);
      if (end.equals(cur)) {
        res.add(new ArrayList<String>(solution));
      } else {
        for (String next : nodeNeighbors.get(cur)) {
          if (distance.get(next) == distance.get(cur) + 1) {
            dfs(next, end, dict, nodeNeighbors, distance, solution, res);
          }
        }
      }
      solution.remove(solution.size() - 1);
    }
  }
}

What to improve

  • for some problem, you have to first use bfs and then use dfs to find solution.