题目
字典 wordList 中从单词 beginWord 和 endWord 的 转换序列 是一个按下述规格形成的序列:
序列中第一个单词是 beginWord 。
序列中最后一个单词是 endWord 。
每次转换只能改变一个字母。
转换过程中的中间单词必须是字典 wordList 中的单词。
给你两个单词 beginWord 和 endWord 和一个字典 wordList ,找到从 beginWord 到 endWord 的 最短转换序列 中的 单词数目 。如果不存在这样的转换序列,返回 0。
示例 1:
输入:beginWord = “hit”, endWord = “cog”, wordList = [“hot”,“dot”,“dog”,“lot”,“log”,“cog”]
输出:5
解释:一个最短转换序列是 “hit” -> “hot” -> “dot” -> “dog” -> “cog”, 返回它的长度 5。
示例 2:
输入:beginWord = “hit”, endWord = “cog”, wordList = [“hot”,“dot”,“dog”,“lot”,“log”]
输出:0
解释:endWord “cog” 不在字典中,所以无法进行转换。
- 方法一:单队列的BFS
class Solution {
public:
//节点存储当前单词以及它在BFS中对应的level
struct node{
string word;
int level;
};
int ladderLength(string beginWord, string endWord, vector<string>& wordList) {
unordered_set<string> s;//存储字典
for(auto &word:wordList){//遍历单词列表,初始化字典
s.insert(word);
}
//队列辅助实现BFS
queue<node> q;
q.push((node){beginWord,1});//初始单词入队列
while(!q.empty()){
node temp = q.front();//获取当前的队首节点
q.pop();
if(temp.word==endWord) return temp.level;//如果当前节点的word等于endword,表示找到最短序列,返回它的level
//给当前单词的每一个字母
for(int i=0; i<temp.word.size(); ++i){
string t = temp.word;
//尝试改为26个字母中的任意一个
for(char c='a'; c<='z'; ++c){
t[i] = c;
if(s.count(t)==1){//如果修改后的字母在字典中,表示找到了一种转换方法,将当前单词入队列
q.push((node){t,temp.level+1});
s.erase(t);//当前节点已经入队列,为了避免多次入队列造成循环,将字典中的节点删除
}
}
}
}
return 0;
}
};
- 时间复杂度O(n)
- 空间复杂度O(n)
- 思路
- 找到从起始单词到最终单词的最短转换序列中的单词序列,类比BFS遍历图时的层数
- 用unordered_map存储字典中的所有单词,方便查找
- 使用队列辅助实现BFS
- 先将起始单词入队列
- 队列非空时,继续while循环,依次访问队列中的每一个节点。循环内先将队首元素出队列,从它的第一个字母开始,尝试修改每一个字母,遍历26个字母的每一种情况。对每一个单词,都查找unordered_map,判断该单词是否在字典中,如果是,则将该单词入队列,初始化时它的level为当前level+1,表示找到了BFS进入了下一层。为了防止同一个节点重复入队列导致循环,将unordered_map中的找到的当前元素删除。
- 在访问队列中的元素时,如果该节点的word是endword,表示有转换序列,返回此时的level。如果直到while循环结束也没有返回,表示没有符合要求的转换序列,返回0.