Solution: Word Search II

Let's solve the Word Search II problem using the Trie pattern.

Statement#

You are given a list of strings that you need to find in a 2D grid of letters such that the string can be constructed from letters in sequentially adjacent cells. The cells are considered sequentially adjacent when they are neighbors to each other either horizontally or vertically. The solution should return a list containing the strings from the input list that were found in the grid.

Constraints:

  • 11 \leq rows, columns 12\leq 12
  • 11 \leq words.length 3×103\leq 3 \times {10^3}
  • 11 \leq words[i].length 10\leq 10
  • grid[i][j] is an uppercase English letter.
  • words[i] consists of uppercase English letters.
  • All the strings are unique.

Note: The order of the strings in the output does not matter.

Solution#

By using backtracking, we can explore different paths in the grid to search the string. We can backtrack and explore another path if a character is not a part of the search string. However, backtracking alone is an inefficient way to solve the problem, since several paths have to be explored to search for the input string.

By using the trie data structure, we can reduce this exploration or search space in a way that results in a decrease in the time complexity:

  • First, we’ll construct the Trie using all the strings in the list. This will be used to match prefixes.

  • Next, we’ll loop over all the cells in the grid and check if any string from the list starts from the letter that matches the letter of the cell.

  • Once an letter is matched, we use depth-first-search recursively to explore all four possible neighboring directions.

  • If all the letters of the string are found in the grid. This string is stored in the output result array.

  • We continue the steps of all our input strings.

Created with Fabric.js 3.6.6 m a a o z i a f h n l s e g m w Trie m t a i n h e Given the 2D grid and input list["main", "the"], add the input liststrings in the Trie.

1 of 50

Created with Fabric.js 3.6.6 m a a o z i a f h n l s e g m w Trie m t a i n h e Since "m" matches one of the childof the Tries root, we search for the sequentially adjacent letters.

2 of 50

Created with Fabric.js 3.6.6 m a a o z i a f h n l s e g m w Trie m t a i n h e Since "a" is the next characterin the Trie, using DFS, we movein this direction.

3 of 50

Created with Fabric.js 3.6.6 m a a o z i a f h n l s e g m w Trie m t a i n h e Since "i" is the next characterin the Trie, using DFS, we movein this direction.

4 of 50

Created with Fabric.js 3.6.6 m a a o z i a f h n l s e g m w Trie m t a i n h e Since "n" is the end of the string,we'll add the word "main" to theresult.

5 of 50

Created with Fabric.js 3.6.6 m a a o z i a f h n l s e g m w Trie m t a i n h e Now, restore the grid. Search "t"in the grid, since it is the first character of the string.

6 of 50

Created with Fabric.js 3.6.6 m a a o z i a f h n l s e g m w Trie m t a i n h e Since "t" is not in the grid, we don'tneed to search further and returnthe result ["main"].

7 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Given the 2D grid and input list["street", "streetcar", "string", "sting","ring", "racecar"], add the input listsin the Trie.

8 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "s" matches one of the childof the Tries root, we search for the sequentially adjacent letters.

9 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "t" is the next characterin the Trie, using DFS, we movein this direction.

10 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "r" is the next characterin the Trie, using DFS, we movein this direction.

11 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "e" is the next characterin the Trie, using DFS, we movein this direction.

12 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "e" is the next characterin the Trie, using DFS, we movein this direction.

13 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "t" is the end of the string,we'll add the word "street"to the result.

14 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Now, restore the grid. Search "s"in the grid, since it is the firstcharacter of the string.

15 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "s" matches one of the childof the Trie's root, we search for the sequentially adjacent letters.

16 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "t" is the next characterin the Trie, using DFS, we movein this direction.

17 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "r" is the next characterin the Trie, using DFS, we movein this direction.

18 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "e" is the next characterin the Trie, using DFS, we movein this direction.

19 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "e" is the next characterin the Trie, using DFS, we movein this direction.

20 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "t" is the next characterin the Trie, using DFS, we movein this direction.

21 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "c" is the next characterin the Trie, using DFS, we movein this direction.

22 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "a" is the next characterin the Trie, using DFS, we movein this direction.

23 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "r" is the end of the string,we'll add the word "streetcar"to the result.

24 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Now, restore the grid. Search "s"in the grid, since it is the firstcharacter of the string.

25 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "s" matches one of the childof the Trie's root, we search for the sequentially adjacent letters.

26 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "t" is the next characterin the Trie, using DFS, we movein this direction.

27 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "r" is the next characterin the Trie, using DFS, we movein this direction.

28 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "i" is the next characterin the Trie, using DFS, we movein this direction.

29 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "n" is the next characterin the Trie, using DFS, we movein this direction.

30 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "g" is the end of the string,we'll add the word "string" tothe result.

31 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Now, restore the grid. Search "s"in the grid, since it is the firstcharacter of the string.

32 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "s" matches one of the childof the Trie's root, we search for the sequentially adjacent letters.

33 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "t" is the next characterin the Trie, using DFS, we movein this direction.

34 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "i" is the next characterin the Trie, using DFS, we movein this direction.

35 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "n" is the next characterin the Trie, using DFS, we movein this direction.

36 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r As "g" is the end of the string,we'll add the word "sting" tothe result.

37 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Now, restore the grid. Search "r"in the grid, since it is the firstcharacter of the string.

38 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "r" matches one of the childof the Trie's root, we search for the sequentially adjacent letters.

39 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "i" is the next characterin the Trie, using DFS, we movein this direction.

40 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "n" is the next characterin the Trie, using DFS, we movein this direction.

41 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "g" is the end of the string,we'll add the word "ring" to theresult.

42 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Now, restore the grid. Search "r"in the grid, since it is the firstcharacter of the string.

43 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "r" matches one of the childof the Trie's root, we search for the sequentially adjacent letters.

44 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "a" is the next characterin the Trie, using DFS, we movein this direction.

45 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "c" is the next characterin the Trie, using DFS, we movein this direction.

46 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "e" is the next characterin the Trie, using DFS, we movein this direction.

47 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "c" is the next characterin the Trie, using DFS, we movein this direction.

48 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "a" is the next characterin the Trie, using DFS, we movein this direction.

49 of 50

Created with Fabric.js 3.6.6 s t r a c i r e e e n g i t c i t s r a Trie s r t r i e i e t c a r n g n g i a n g c e c a r Since "r" is the end of the string,we'll add the word "racecar" tothe result.

50 of 50

Let’s look at the code for this solution below:

main.py
trie.py
trie_node.py
Word Search II

Time complexity#

The time complexity will be O(n3l)O(n*3^l), where nn is equal to rows*columns, and ll is the length of the longest string in the list. The factor 3l3^l means that, in the dfs() function, we have four directions to explore initially, but only three choices remain in each cell because one has already been explored. In the worst case, none of the strings will have the same prefix, so we cannot skip any string from the list.

Space complexity#

In the worst case, none of the strings will have the same prefix, so the space used by the trie data structure will be O(m)O(m), where mm is the total count of all the characters in all the strings present in the input list.

If there is a string of length equal to the length of the grid and the elements on the grid are arranged in such a way that the string can be traced, then the space acquired on the stack to store the recursive calls of dfs() function would be the size of the grid, which is O(n)O(n).

Hence, the overall space complexity of the solution will be O(m+n)O(m+n).

Word Search II

Custom Data Structures: Introduction