There is a new alien language that uses the English alphabet. However, the order among the letters is unknown to you.
You are given a list of strings words from the alien language's dictionary, where the strings in words are sorted lexicographically by the rules of this new language.
Return a string of the unique letters in the new alien language sorted in lexicographically increasing order by the new language's rules. If there is no solution, return "". If there are multiple solutions, return any of them.
A string s is lexicographically smaller than a string t if at the first letter where they differ, the letter in s comes before the letter in t in the alien language. If the first min(s.length, t.length) letters are the same, then s is smaller if and only if s.length < t.length.
Example 1:
Input: words = ["wrt","wrf","er","ett","rftt"]
Output: "wertf"
Example 2:
Input: words = ["z","x"]
Output: "zx"
Example 3:
Input: words = ["z","x","z"]
Output: ""
Explanation: The order is invalid, so return "".
Constraints:
1 <= words.length <= 100
1 <= words[i].length <= 100
words[i] consists of only lowercase English letters.
// neighbor represent the list of next nodes Map<Character, Set<Character>> neighbors = new HashMap<>(); // indegrees represent how many dependent nodes Map<Character, Integer> indegrees = new HashMap<>();
// initialize neighbors and indegree for (int i = 0; i < words.length; i++){ for(int j = 0; j < words[i].length(); j++){ char ch = words[i].charAt(j); if (!neighbors.containsKey(ch)) { neighbors.put(ch, new HashSet<Character>()); } indegrees.put(ch, 0); } }
// 查看所有相近 chars,放入neighbors的 HashSet for (int i = 0; i + 1 < words.length; i++){ int p = 0; int len = Math.min(words[i].length(), words[i + 1].length()); while (p < len) { char first = words[i].charAt(p); char second = words[i + 1].charAt(p); if (first != second) { // add first -> second to neighbors and indegrees neighbors.get(first).add(second); break; } p++; } if (p == len && words[i].compareTo(words[i + 1]) > 0) { // the {"abc", "ab"} case, directly return empty return ""; } }
// build indegrees for (char key : neighbors.keySet()) { for (char inChar: neighbors.get(key)) { indegrees.put(inChar, indegrees.get(inChar) + 1); } }
// start the toplogical sorting PriorityQueue<Character> queue = new PriorityQueue<>(); for (char key : indegrees.keySet()){ if(indegrees.get(key) == 0){ queue.add(key); } }
StringBuilder sb = new StringBuilder(); while (!queue.isEmpty()) { char cur = queue.remove(); sb.append(cur); for(char neighbor : neighbors.get(cur)){ int new_indegree = indegrees.get(neighbor) - 1; if(new_indegree == 0){ queue.add(neighbor); } indegrees.put(neighbor, new_indegree); } }