leetcode--Word Ladder II

本文介绍了一种寻找两个单词间最短转换路径的算法,通过广度优先搜索实现,确保每次只改变一个字母,并且每一步骤都在给定字典中。

Given two words (start and end), and a dictionary, find all shortest transformation sequence(s) from start to end, such that:

  1. Only one letter can be changed at a time
  2. Each intermediate word must exist in the dictionary

For example,

Given:
start = "hit"
end = "cog"
dict = ["hot","dot","dog","lot","log"]

Return

  [
    ["hit","hot","dot","dog","cog"],
    ["hit","hot","lot","log","cog"]
  ]

Note:

  • All words have the same length.

  • All words contain only lowercase alphabetic characters.


题意:

给定两个单词(起始单词和终止单词),和一本字典。找到从起始单词到终止单词的最短转换路径。

要求:每次只能转换一个字母

    每个转换后的单词都必须在字典里面

例如”hit“为起始单词,"cog"为终止单词

字典为{"hot","dot","dog","lot","log"}

其中一条最短路径为"hot","dot","dog","lot","log"

Note: 所有的单词长度都相同

            单词中都是小写字母

分类:字符串,图,广度搜索


解法1:基本思路和Word Ladder相同。关键在于存储所有的路径。

为了存储路径,需要为每个节点增加一个pre指针指向它的前一个节点。

另外注意,在遍历某一层的时候,如果节点a找到了下一个节点next,不要马上将其从字典里面删除

因为这一层的某个节点b,也可能下一层节点也是它next。

如果你删除了,那么这个b就找不到next了,而这也是一条路径。

所以合适的做法是,当这一层全部遍历完,我们才删除已经访问过的下一层节点。

这就需要我们记录下一层访问过的节点,在代码里面我们用visited来记录。

[java]  view plain  copy
  1. public class Solution {  
  2.     public List<List<String>> findLadders(String start, String end, Set<String> dict) {  
  3.         class WordNode{//这个类用于标记单词位于广度搜索的哪一层    
  4.             String word;    
  5.             int step;    
  6.             WordNode pre;  
  7.             public WordNode(String word, int step,WordNode pre) {    
  8.                 this.word = word;    
  9.                 this.step = step;    
  10.                 this.pre = pre;  
  11.             }                           
  12.         }           
  13.           
  14.         ArrayList<List<String>> res = new ArrayList<List<String>>();  
  15.         dict.add(end);  
  16.         HashSet<String> visited = new HashSet<String>();//用于标记下一层已经访问过的节点  
  17.         HashSet<String> unvisited = new HashSet<String>();//用于标记还没有访问的节点  
  18.         unvisited.addAll(dict);  
  19.         Queue<WordNode> queue = new LinkedList<WordNode>();//队列    
  20.         queue.add(new WordNode(start,1,null));//起始单词加入队列    
  21.         int preNumSteps = 0;//队列中上一个节点所在的层  
  22.         int minStep = 0;//最短路径长度  
  23.           
  24.         while(!queue.isEmpty()){  
  25.             WordNode cur = queue.poll();    
  26.             int currNumSteps = cur.step;  
  27.             if(cur.word.equals(end)){//判断是否是终止单词  
  28.                 if(minStep == 0){//判断是否第一次找到最短路径  
  29.                     minStep = cur.step;//更新最短路径  
  30.                 }  
  31.    
  32.                 if(cur.step == minStep && minStep !=0){//如果是最短路径,添加结果  
  33.                     ArrayList<String> t = new ArrayList<String>();  
  34.                     t.add(cur.word);  
  35.                     while(cur.pre !=null){  
  36.                         t.add(0, cur.pre.word);  
  37.                         cur = cur.pre;  
  38.                     }  
  39.                     res.add(t);  
  40.                     continue;  
  41.                 }  
  42.             }    
  43.             if(preNumSteps < currNumSteps){//说明这是下一层的开始,这时可以清空所有的访问过的节点  
  44.                 unvisited.removeAll(visited);  
  45.             }  
  46.             preNumSteps = currNumSteps;  
  47.             char[] arr = cur.word.toCharArray();    
  48.             for(int i=0;i<arr.length;i++){//遍历单词每个字母    
  49.                 for(int c='a';c<='z';c++){//更换    
  50.                     char temp = arr[i];    
  51.                     if(c!=temp){    
  52.                         arr[i] = (char) c;    
  53.                     }  
  54.                     String str = new String(arr);    
  55.                     if(unvisited.contains(str)){//判断变换后的单词是不是在字典里面    
  56.                         queue.add(new WordNode(str, currNumSteps+1,cur));//加入队列    
  57.                         visited.add(str);//从字典中移除    
  58.                     }    
  59.                     arr[i] = temp;//恢复原单词    
  60.                 }    
  61.             }    
  62.         }  
  63.         return res;  
  64.     }  
  65. }  

原文链接http://blog.youkuaiyun.com/crazy__chen/article/details/47337157

已经博主授权,源码转载自 https://pan.quark.cn/s/a4b39357ea24 常见问题解答 网页打开速度慢或者打不开网页? 受到多种因素的影响,对于非会员用户我们无法提供最优质的服务。 如果您希望得到最棒的体验,请至大会员页面("右上角菜单 → 大会员")根据说明操作。 请注意:受制于国际网络的诸多不确定性,我们无法对任何服务的可靠性做出任何保证。 如果出现了网络连接相关的问题,我们建议您先等待一段时间,之后再重试。 如果您在重试后发现问题仍然存在,请联系我们,并说明网络问题持续的时间。 图片下载后无法找到? 打开"右上角菜单 → 更多 → 修改下载路径",在弹出的对话框中可以看到当前图片的保存路径。 此外,由于网络因素,在保存图片之后,等待屏幕下方出现"已保存到..."后,才能在本地找到图片。 如何更改图片保存的目录? 请参见"右上角菜单 → 更多 → 修改下载路径"。 翻页不方便? 在点进某个图片后,通过在图片上向左或向右滑动,即可翻页查看下一个作品。 如何保存原图/导出动图? 长按图片/动图,在弹出的菜单中选择保存/导出即可。 输入账号密码后出现"进行人机身份验证"? 此为pixiv登陆时的验证码,请按照要求点击方框或图片。 在pxvr中注册pixiv账号后,收到验证邮件,无法访问邮件中的验证链接? 请复制邮件中的链接,打开pxvr中的"右上角菜单 → 输入地址"进行访问。 能否自动将页面内容翻译为汉语? 很抱歉,pxvr暂不提供语言翻译服务。 图片下载类型是否可以选择? 能否批量下载/批量管理下载? 已支持批量下载多图作品中的所有原图:找到一个多图作品,进入详情页面后,点击图片进入多图浏览模式,长按任意一张图片即可看到批量下载选项。 关于上述其他功能,我们...
考虑局部遮阴的光伏PSO-MPPT控制模型(Simulink仿真实现)内容概要:本文介绍了基于Simulink仿真实现的考虑局部遮阴的光伏PSO-MPPT控制模型,旨在通过粒子群优化(PSO)算法解决光伏发电系统在局部阴影条件下最大功率点跟踪(MPPT)的效率问题。文档不仅提供了该模型的技术实现方法,还列举了大量相关的MATLAB/Simulink仿真资源,涵盖电力系统、智能优化算法、机器学习、路径规划、信号处理等多个科研方向,适用于复现高水平期刊论文和开展创新性研究。文中强调科研需逻辑缜密、善于借力,并提倡结合实际仿真与理论分析以提升研究深度。 适合人群:具备一定电力电子、自动控制或新能源背景,熟悉MATLAB/Simulink环境,从事光伏系统优化、智能算法应用或相关领域研究的研发人员及硕博研究生。 使用场景及目标:①研究局部遮阴下光伏系统MPPT控制策略的性能提升;②利用PSO等智能优化算法解决非线性、多峰值优化问题;③复现SCI/EI级别论文中的MPPT控制模型;④开展光伏系统建模与仿真教学或项目开发。 阅读建议:建议读者结合文档提供的网盘资源,下载完整代码与模型文件,按照目录顺序逐步学习,重点理解PSO算法在MPPT中的应用机制,并通过修改参数、对比实验等方式深入掌握仿真细节,提升工程实践与科研创新能力。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值