zoukankan      html  css  js  c++  java
  • leetcode——126.单词接龙II

    自己做的时候老是想着走捷径,其实代码不都是从最复杂最直白的思路开始,然后一步一步进行优化的吗,所以还是不能心急。

    这道题我这次没做出来,踏踏实实来。

    public List<List<String>> findLadders(String beginWord, String endWord, List<String> wordList) {
            // 先将 wordList 放到哈希表里,便于判断某个单词是否在 wordList 里
            Set<String> wordSet = new HashSet<>(wordList);
            List<List<String>> res = new ArrayList<>();
            if (wordSet.size() == 0 || !wordSet.contains(endWord)) {
                return res;
            }
    
            // 第 1 步:使用广度优先遍历得到后继结点列表 successors
            // key:字符串,value:广度优先遍历过程中 key 的后继结点列表
            Map<String, Set<String>> successors = new HashMap<>();
            boolean found = bfs(beginWord, endWord, wordSet, successors);
            if (!found) {
                return res;
            }
    
            // 第 2 步:基于后继结点列表 successors ,使用回溯算法得到所有最短路径列表
            Deque<String> path = new ArrayDeque<>();
            path.addLast(beginWord);
            dfs(beginWord, endWord, successors, path, res);
            return res;
        }
    
        private boolean bfs(String beginWord, String endWord, Set<String> wordSet,
                            Map<String, Set<String>> successors) {
            Queue<String> queue = new LinkedList<>();
            queue.offer(beginWord);
    
            // 标准写法,记录方法问过的单词
            Set<String> visited = new HashSet<>();
            visited.add(beginWord);
    
            boolean found = false;
            int wordLen = beginWord.length();
            // 当前层访问过的结点,当前层全部遍历完成以后,再添加到总的 visited 集合里
            Set<String> nextLevelVisited = new HashSet<>();
            while (!queue.isEmpty()) {
                int currentSize = queue.size();
                for (int i = 0; i < currentSize; i++) {
                    String currentWord = queue.poll();
                    char[] charArray = currentWord.toCharArray();
                    for (int j = 0; j < wordLen; j++) {
                        char originChar = charArray[j];
                        for (char k = 'a'; k <= 'z'; k++) {
                            if (charArray[j] == k) {
                                continue;
                            }
                            charArray[j] = k;
                            String nextWord = new String(charArray);
                            if (wordSet.contains(nextWord)) {
                                if (!visited.contains(nextWord)) {
                                    if (nextWord.equals(endWord)) {
                                        found = true;
                                    }
    
                                    // 避免下层元素重复加入队列,这里感谢 https://leetcode-cn.com/u/zhao-da-ming/ 优化了这个逻辑
                                    if (!nextLevelVisited.contains(nextWord)) {
                                        queue.offer(nextWord);
                                        nextLevelVisited.add(nextWord);
                                    }
    
                                    // 维护 successors 的定义
                                    successors.computeIfAbsent(currentWord, a -> new HashSet<>());
                                    successors.get(currentWord).add(nextWord);
                                }
                            }
                        }
                        charArray[j] = originChar;
                    }
                }
    
                if (found) {
                    break;
                }
                visited.addAll(nextLevelVisited);
                nextLevelVisited.clear();
            }
            return found;
        }
    
        private void dfs(String beginWord, String endWord,
                         Map<String, Set<String>> successors,
                         Deque<String> path, List<List<String>> res) {
            if (beginWord.equals(endWord)) {
                res.add(new ArrayList<>(path));
                return;
            }
    
            if (!successors.containsKey(beginWord)) {
                return;
            }
    
            Set<String> successorWords = successors.get(beginWord);
            for (String nextWord : successorWords) {
                path.addLast(nextWord);
                dfs(nextWord, endWord, successors, path, res);
                path.removeLast();
            }
        }

    ——2020.8.3

    我的前方是万里征途,星辰大海!!
  • 相关阅读:
    javascript数据类型判断
    Week04面向对象设计与继承
    201621044079《Java程序设计》第1周学习总结
    201621044079 week05继承、多态、抽象类与接口
    201621044079《Java程序设计》第二周学习总结
    Week03面向对象入门
    201621044079WEEK06接口、内部类
    202020211 20209320 《Linux内核原理与分析》第一周作业
    第二天:PowerShell别名
    第一天:powershell外部命令
  • 原文地址:https://www.cnblogs.com/taoyuxin/p/13425667.html
Copyright © 2011-2022 走看看