一刷308-剑指 Offer II 114. 外星文字典(h)

题目:
现有一种使用英语字母的外星文语言,这门语言的字母顺序与英语顺序不同。

给定一个字符串列表 words ,作为这门语言的词典,
words 中的字符串已经 按这门新语言的字母顺序进行了排序 。

请你根据该词典还原出此语言中已知的字母顺序,并 按字母递增顺序 排列。
若不存在合法字母顺序,返回 "" 。
若存在多种可能的合法字母顺序,返回其中 任意一种 顺序即可。

字符串 s 字典顺序小于 字符串 t 有两种情况:

在第一个不同字母处,如果 s 中的字母在这门外星语言的字母顺序中位于 t 中字母之前,
那么 s 的字典顺序小于 t 。

如果前面 min(s.length, t.length) 字母都相同,
那么 s.length < t.length 时,s 的字典顺序也小于 t 。
-------------------
示例 1:
输入:words = ["wrt","wrf","er","ett","rftt"]
输出:"wertf"
示例 2:

输入:words = ["z","x"]
输出:"zx"
示例 3:

输入:words = ["z","x","z"]
输出:""
解释:不存在合法字母顺序,因此返回 "" 。
 
提示:
1 <= words.length <= 100
1 <= words[i].length <= 100
----------------
思路:
根据题意,需要从给出的字符串数组中找出出现字母之间的顺序。
各字符串中在字符串数组words中已经是按递增顺序排列,
我们可以通过比较数组中两两相邻的字符串中的对应字符得出字符比较大小的结果。
例如,给定字符串数组words = ["wrt","wrf","er","ett","rftt"].

因为words[0] < words[1],可知't' < 'f';
words[1] < words[2],可知‘w' < 'e';
words[2] < words[3],可知'r' < 't';后续比较规律相同。
对于此类题应该通过具体的例子抽象出相应规律。
另外通过提交我发现有一个特殊输入words = ["abc","ab"],后一个字符串是前一个字符串的前缀,
即前一个字符串长度大于后一个,这就违背了题目说明的字符串数组是递增排列,这就是一个不合法的无效输入,
应返回空字符串"".

通过以上分析,我们可以把这个问题抽象成一个图遍历问题,字符表示图节点,
排在靠前顺序的字符有一条指向靠后顺序字符的有向边,而最靠前的字符表示入度为0的节点,
通过广度优先遍历每次把所有入度为0的节点放进队列,
在遍历过程中队列中的节点顺序就是字符之间的相对顺序(之一,因为由给定的条件无法判断唯一顺序,
同时拓扑排序如果成功那么也不一定有唯一解)。

明白此问题的算法思想以后,就转换成了建图再BFS遍历的过程。用一个HashMap来表示图节点之间的关系,
key表示该字符,value表示顺序排在该字符之后的所有字符,
即该字符key有一条指向value中各个字符的有向边,又因为字符不用重复保存,
只需一条边表示两个字符之间的关系,我们就用HashSet做value表示该字符集合。
这样建立好字符之间的顺序关系以后,由于需要使用拓扑排序,
我们还需要用一个数组或者哈希表保存每个字符的入度数量,因为此题输入只有小写字母,
只用一个空间为26的数组保存即可。
--------------------
class Solution {
    public String alienOrder(String[] words) {
        Map<Character, Set<Character>> map = new HashMap<>();
        int[] inDegree = new int[26];
        Queue<Character> queue = new LinkedList<>();
        StringBuilder sb = new StringBuilder();
        for (String word : words) {//把出现的字符保存在图HashMap中,每个不重复字符对应
            for (char ch : word.toCharArray()) {//一个表示有序的有向边
                map.putIfAbsent(ch, new HashSet<>());
            }
        }
        for (int i = 1; i < words.length; i++) {//两两比较相邻字符串之间的关系
            String w1 = words[i - 1];
            String w2 = words[i];
            if (checkPre(w1, w2) && !w1.equals(w2)) return "";//检查为不合法输入
            for (int j = 0; j < Math.min(w1.length(), w2.length()); j++) {
                char c1 = w1.charAt(j);
                char c2 = w2.charAt(j);
                if (c1 != c2) {//找到不同的字符才说明有顺序关系,把前一个字符指向后一个字符
                    if (!map.get(c1).contains(c2)) {//同时后一字符的入度 + 1
                        map.get(c1).add(c2);
                        inDegree[c2 - 'a']++;
                    }
                    break;
                }
            }
        }
        for (char ch : map.keySet()) {//把所有入度为0的字符先加入队列,准备拓扑排序
                if (inDegree[ch - 'a'] == 0) queue.offer(ch);
        }
        while (!queue.isEmpty()) {
            char node = queue.poll();//从队列出来的这个字符肯定是入度为0,可以确定它的顺序
            sb.append(node);//就把它加进字符顺序的结果里
            for (char next : map.get(node)) {//取出当前出来的节点以后,该节点所有的出度
                inDegree[next - 'a']--;//(有向边指向所有节点的节点入度)都要减一
                if (inDegree[next - 'a'] == 0) queue.offer(next);//若有入度为0的节点字符出现
            }//把它加进队列准备之后的拓扑排序遍历
        }
        return sb.length() == map.size() ? sb.toString() : "";//否则不成功,不存在合法字母顺序
    }//若结果集里的字符数量和图中所有节点数量相同,说明拓扑排序成功,返回结果集
    private boolean checkPre(String s1, String s2) {
        int m = s1.length();
        int n = s2.length();
        if (m < n) return false;
        int i = 0;
        int j = 0;
        while (i < m && j < n) {
            if (s2.charAt(j) != s1.charAt(i)) return false;
            i++;
            j++;
        }
        return true;
    }
}

LC

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值