Solution: Alien Dictionary
Let's solve the Alien Dictionary problem using the Topological Sort pattern.
Statement
In this challenge, you are given a list of words written in an alien language, where the words are sorted lexicographically by the rules of this language. Surprisingly, the aliens also use English lowercase letters, but possibly in a different order.
Given a list of words written in the alien language, you have to return a string of unique letters sorted in the lexicographical order of the alien language as derived from the list of words.
If there’s no solution, that is, no valid lexicographical ordering, you can return an empty string.
Note: The lexicographic order of a given language is defined by the order in which the letters of its alphabet appear. In English, the letter “n” appears before the letter “r” in the alphabet. As a result, in two words that are the same up to the point where one features “n” and the other features “r,” the former is considered the lexicographically smaller word of the two. For this reason, “ban” is considered lexicographically smaller than “bar.”
Similarly, if an input contains words followed by their prefix, such as “educated” and then “educate,” these cases will never result in a valid alphabet because in a valid alphabet, prefixes are always first.
Constraints:
-
words.length
-
words[i].length
- All characters in
words[i]
are English lowercase letters.
Solution
So far, you’ve probably brainstormed some approaches and have an idea of how to solve this problem. Let’s explore some of these approaches and figure out which one to follow based on considerations such as time complexity and any implementation constraints.
Naive approach
The naive approach is to generate all possible orders of alphabets in the alien language and then iterate over them, character by character, to select the ones that satisfy the dictionary dependencies. So, there’d be O(u!) permutations, where is the number of unique alphabets in the alien language, and for each permutation, we’d have to check if it’s a valid partial order. That requires comparing against the dictionary words repeatedly.
This is very expensive since there are an exponential number of possible orders () and only a handful of valid ones. On top of that, there’d be additional effort to compare them against the dictionary. The time complexity for this approach is . The space complexity is
Optimized approach using topological sort
We can solve this problem using the topological sort pattern. Topological sort is used to find a linear ordering of elements that have dependencies on or priority over each other. For example, if is dependent on or has priority over , then is listed before in topological order.
Using the list of words, we identify the relative precedence order of the letters in the words and generate a graph to represent this ordering. To traverse a graph, we can use breadth-first search to find the letters’ order.
We can essentially map this problem to a graph problem, but before exploring the exact details of the solution, there are a few things that we need to keep in mind:
-
The letters within a word don’t tell us anything about the relative order. For example, the word “educative” in the list doesn’t tell us that the letter “e” is before the letter “d.”
-
The input can contain words followed by their prefix, such as “educated” and then “educate.” These cases will never result in a valid alphabet because in a valid alphabet, prefixes are always first. We need to make sure our solution detects these cases correctly.
-
There can be more than one valid alphabet ordering. It’s fine for our algorithm to return any one of them.
-
The output dictionary must contain all unique letters within the words list, including those that could be in any position within the ordering. It shouldn’t contain any additional letters that weren’t in the input.
Note: In the following section, we will gradually build the solution. Alternatively, you can skip straight to just the code.
Step-by-step solution construction
For the graph problem, we can break this particular problem into three parts:
-
Extract the necessary information to identify the dependency rules from the words. For example, in the words [“patterns”, “interview”], the letter “p” comes before “i.”
-
With the gathered information, we can put these dependency rules into a directed graph with the letters as nodes and the dependencies (order) as the edges.
-
Lastly, we can sort the graph nodes topologically to generate the letter ordering (dictionary).
Let’s look at each part in more depth.
Part 1: Identifying the dependencies
Let’s start with example words and observe the initial ordering through simple reasoning:
["mzosr", "mqov", "xxsvq", "xazv", "xazau", "xaqu", "suvzu", "suvxq", "suam", "suax", "rom", "rwx", "rwv"]
As in the English language dictionary, where all the words starting with “a” come at the start followed by the words starting with “b,” “c,” “d,” and so on, we can expect the first letters of each word to be in alphabetical order.
["m", "m", "x", "x", "x", "x", "s", "s", "s", "s", "r", "r", "r"]
Removing the duplicates, we get the following:
["m", "x", "s", "r"]
Following the intuition explained above, we can assume that the first letters in the messages are in alphabetical order:
Looking at the letters above, we know the relative order of these letters, but we don’t know how these letters fit in with the rest of the letters. To get more information, we need to look further into our English dictionary analogy. The word “dirt” comes before “dorm.” This is because we look at the second letter when the first letter is the same. In this case, “i” comes before “o” in the alphabet.
We can apply the same logic to our alien words and look at the first two words, “mzsor” and “mqov.” As the first letter is the same in both words, we look at the second letter. The first word has “z,” and the second one has “q.” Therefore, we can safely say that “z” comes before “q” in this alien language. We now have two fragments of the letter order:
Note: Notice that we didn’t mention rules such as “m -> a”. This is fine because we can derive this relation from “m -> x”, “x -> a”.
This is it for the first part. Let’s put the pieces that we have in place.
Part 2: Representing the dependencies
We now have a set of relations mentioning the relative order of the pairs of letters:
["z -> q", "m -> x", "x -> a", "x -> v", "x -> s", "z -> x", "v -> a", "s -> r", "o -> w"]
Now the question arises, how can we put these relations together? It might be tempting to start chaining all these together. Let’s look at a few possible chains:
We can observe from our chains above that some letters might appear in more than one chain, and putting the chains into the output list one after the other won’t work. Some of the letters might be duplicated and would result in an invalid ordering. Let’s try to visualize the relations better with the help of a graph. The nodes are the letters, and an edge between two letters, “x” and “y” represents that “x” is before “y” in the alien words.
Part 3: Generating the dictionary
As we can see from the graph, four of the letters have no incoming arrows. This means that there are no letters that have to come before any of these four.
Remember: There could be multiple valid dictionaries, and if there are, then it’s fine for us to return any of them.
Therefore, a valid start to the ordering we return would be as follows:
["o", "m", "u", "z"]
We can now remove these letters and edges from the graph because any other letters that required them first will now have this requirement satisfied.
There are now three new letters on this new graph that have no in arrows. We can add these to our output list.
["o", "m", "u", "z", "x", "q", "w"]
Again, we can remove these from the graph.
Then, we add the two new letters with no in arrows.
["o", "m", "u", "z", "x", "q", "w", "v", "s"]
This leaves the following graph:
We can place the final two letters in our output list and return the ordering:
["o", "m", "u", "z", "x", "q", "w", "v", "s", "a", "r"]
Let’s now review how we can implement this approach.
Identifying the dependencies and representing them in the form of ...