面试算法108:单词演变

news/2024/7/24 10:49:13 标签: 面试, 算法, java

题目

输入两个长度相同但内容不同的单词(beginWord和endWord)和一个单词列表,求从beginWord到endWord的演变序列的最短长度,要求每步只能改变单词中的一个字母,并且演变过程中每步得到的单词都必须在给定的单词列表中。如果不能从beginWord演变到endWord,则返回0。假设所有单词只包含英文小写字母。
例如,如果beginWord为"hit",endWord为"cog",单词列表为[“hot”,“dot”,“dog”,“lot”,“log”,“cog”],则演变序列的最短长度为5,一个可行的演变序列为"hit"→"hot"→"dot"→"dog"→"cog"。

分析

应用图相关算法的前提是找出图中的节点和边。这个问题是关于单词的演变的,所以每个单词就是图中的一个节点。如果两个单词能够相互演变(改变一个单词的一个字母能变成另一个单词),那么这两个单词之间有一条边相连。
在这里插入图片描述
为了求得两个节点之间的最短距离,常见的解法是用两个队列实现广度优先搜索算法。一个队列queue1中存放离起始节点距离为d的节点,当从这个队列中取出节点并访问的时候,与队列queue1中节点相邻的节点离起始节点的距离都是d+1,将这些相邻的节点存放到另一个队列queue2中。当队列queue1中的所有节点都访问完毕时,再访问队列queue2中的节点,并将相邻的节点放入queue1中。可以交替使用queue1和queue2这两个队列由近及远地从起始节点开始搜索所有节点。

解: 单向广度优先搜索

java">public class Test {
    public static void main(String[] args) {
        List<String> wordList = Arrays.asList("hot", "dot", "dog", "lot", "log", "cog");
        int result = ladderLength("hit", "cog", wordList);
        System.out.println(result);
    }

    public static int ladderLength(String beginWord, String endWord, List<String> wordList) {
        Queue<String> queue1 = new LinkedList<>();
        Queue<String> queue2 = new LinkedList<>();
        Set<String> notVisited = new HashSet<>(wordList);
        int length = 1;
        queue1.add(beginWord);
        while (!queue1.isEmpty()) {
            String word = queue1.remove();
            if (word.equals(endWord)) {
                return length;
            }

            List<String> neighbors = getNeighbors(word);
            for (String neighbor : neighbors) {
                if (notVisited.contains(neighbor)) {
                    queue2.add(neighbor);
                    notVisited.remove(neighbor);
                }
            }

            if (queue1.isEmpty()) {
                length++;
                queue1 = queue2;
                queue2 = new LinkedList<>();
            }
        }

        return 0;
    }

    private static List<String> getNeighbors(String word) {
        List<String> neighbors = new LinkedList<>();
        char[] chars = word.toCharArray();
        for (int i = 0; i < chars.length; i++) {
            char old = chars[i];

            for (char ch = 'a'; ch <= 'z'; ++ch) {
                if (old != ch) {
                    chars[i] = ch;
                    neighbors.add(new String(chars));
                }
            }

            chars[i] = old;
        }

        return neighbors;
    }

}

http://www.niftyadmin.cn/n/5317423.html

相关文章

【ARM 嵌入式 编译系列 7.3 -- GCC 链接脚本中 NOLOAD 和 GROUP 的详细介绍】

文章目录 NOLOAD 和 GROUP 的详细介绍NOLOAD 关键字GROUP 关键字实际应用案例NOLOAD 和 GROUP 的详细介绍 在使用 arm-none-eabi-gcc 工具链中的链接器脚本时,链接脚本使用链接器命令语言来描述如何生成最终的可执行文件。其中,noload 和 group 是两个用于控制链接过程的关键…

深入解析HubSpot在线客户互动工具:提升客户体验的利器

在数字化时代&#xff0c;客户体验成为企业成功的关键因素之一。HubSpot作为一体化的市场营销、销售和服务平台&#xff0c;其在线客户互动工具扮演着提升客户体验的重要角色。本文将深入探讨HubSpot的在线客户互动工具&#xff0c;包括实时聊天、机器人和社交媒体监控&#xf…

鸿蒙APP和Android的区别

鸿蒙&#xff08;HarmonyOS&#xff09;和Android是两个不同的操作系统&#xff0c;它们有一些区别&#xff0c;包括架构、开发者支持、应用生态和一些设计理念。以下是鸿蒙APP和Android APP之间的一些主要区别&#xff0c;希望对大家有所帮助。北京木奇移动技术有限公司&#…

【HarmonyOS4.0】第七篇-ArkUI系统组件(二)

鸿蒙开发系统组件详细剖析 五、进度条组件 进度条也是UI开发最常用的组件之一&#xff0c;ArkUI开发框架提供了两种类型的进度条&#xff1a; Progress 和LoadingProgress &#xff0c;前者可以精准指定进度&#xff0c;后者表示正在加载的状态&#xff0c;我们接下来对它们分…

Golang 四数相加 leetcode454 map哈希表

四数相加 leetcode454 本题如果直接进行四次for循环&#xff0c;则时间复杂度为O(N^4),超出运行时间限制。 因此我们这里使用两个分别的for循环进行遍历&#xff0c;则时间复杂度为O(N2N2). / 使用两遍for循环 func fourSumCount(nums1 []int, nums2 []int, nums3 []int, num…

魏副业而战:视频号情感副业项目,新手也能放大操作

我是魏哥&#xff0c;与其躺平&#xff0c;不如魏副业而战&#xff01; 视频号推出创作分成计划后&#xff0c;吸引很多创作者入驻。 有些小伙伴赚得盆满钵满&#xff0c;让人好生羡慕。 那么&#xff0c;今天魏哥给大家分享一个视频号创作分成计划副业项目。 我们主要做情感…

数据结构第一弹

简述数据结构&#xff0c;抽象数据结构和数据类型之间的异同。 数据结构&#xff0c;抽象数据结构和数据类型本质上来说是同一概念&#xff0c;数据类型是程序设计中实现了的数据结构&#xff0c;而抽象数据结构是数据类型的进一步抽象和发展&#xff0c;借助数据类型可以在程…

【Docker基础四】Docker安装Nacos

准备 需安装mysql并创建nacos数据库。 没有安装mysql&#xff1f;看这篇文章&#xff1a;【Docker基础二】Docker安装Mysql8-CSDN博客 建表语句地址&#xff1a;建表语句(nacos github) 下载镜像 可用版本&#xff1a;发布 阿里巴巴/NACOS (github.com) # 查看镜像 docke…