1 public ArrayList<ArrayList<String>> findLadders(String start, String end, HashSet<String> dict) {
2
3 HashMap<String, HashSet<String>> neighbours = new HashMap<String, HashSet<String>>();
4
5 dict.add(start);
6 dict.add(end);
7
8 // init adjacent graph
9 for(String str : dict){
10 calcNeighbours(neighbours, str, dict);
11 }
12
13 ArrayList<ArrayList<String>> result = new ArrayList<ArrayList<String>>();
14
15 // BFS search queue
16 LinkedList<Node> queue = new LinkedList<Node>();
17 queue.add(new Node(null, start, 1)); //the root has not parent and its level == 1
18
19 // BFS level
20 int previousLevel = 0;
21
22 // mark which nodes have been visited, to break infinite loop
23 HashMap<String, Integer> visited = new HashMap<String, Integer>();
24 while(!queue.isEmpty()){
25 Node n = queue.pollFirst();
26 if(end.equals(n.str)){
27 // fine one path, check its length, if longer than previous path it‘s valid
28 // otherwise all possible short path have been found, should stop
29 if(previousLevel == 0 || n.level == previousLevel){
30 previousLevel = n.level;
31 findPath(n, result);
32 }else {
33 // all path with length *previousLevel* have been found
34 break;
35 }
36 }else {
37 HashSet<String> set = neighbours.get(n.str);
38
39 if(set == null || set.isEmpty()) continue;
40 // note: I‘m not using simple for(String s: set) here. This is to avoid hashset‘s
41 // current modification exception.
42 ArrayList<String> toRemove = new ArrayList<String>();
43 for (String s : set) {
44
45 // if s has been visited before at a smaller level, there is already a shorter
46 // path from start to s thus we should ignore s so as to break infinite loop; if
47 // on the same level, we still need to put it into queue.
48 if(visited.containsKey(s)){
49 Integer occurLevel = visited.get(s);
50 if(n.level+1 > occurLevel){
51 neighbours.get(s).remove(n.str);
52 toRemove.add(s);
53 continue;
54 }
55 }
56 visited.put(s, n.level+1);
57 queue.add(new Node(n, s, n.level + 1));
58 if(neighbours.containsKey(s))
59 neighbours.get(s).remove(n.str);
60 }
61 for(String s: toRemove){
62 set.remove(s);
63 }
64 }
65 }
66
67 return result;
68 }
69
70 public void findPath(Node n, ArrayList<ArrayList<String>> result){
71 ArrayList<String> path = new ArrayList<String>();
72 Node p = n;
73 while(p != null){
74 path.add(0, p.str);
75 p = p.parent;
76 }
77 result.add(path);
78 }
79
80 /*
81 * complexity: O(26*str.length*dict.size)=O(L*N)
82 */
83 void calcNeighbours(HashMap<String, HashSet<String>> neighbours, String str, HashSet<String> dict) {
84 int length = str.length();
85 char [] chars = str.toCharArray();
86 for (int i = 0; i < length; i++) {
87
88 char old = chars[i];
89 for (char c = ‘a‘; c <= ‘z‘; c++) {
90
91 if (c == old) continue;
92 chars[i] = c;
93 String newstr = new String(chars);
94
95 if (dict.contains(newstr)) {
96 HashSet<String> set = neighbours.get(str);
97 if (set != null) {
98 set.add(newstr);
99 } else {
100 HashSet<String> newset = new HashSet<String>();
101 newset.add(newstr);
102 neighbours.put(str, newset);
103 }
104 }
105 }
106 chars[i] = old;
107 }
108 }
109
110 private class Node {
111 public Node parent; //previous node
112 public String str;
113 public int level;
114 public Node(Node p, String s, int l){
115 parent = p;
116 str = s;
117 level = l;
118 }
119 }