1455 检查单词是否为句中其他单词的前缀——Leetcode天天刷(2022.8.21)【双指针】
文章目录
前言
今天的每日一题,有点简单,一开始想用
字典树
或者KMP
来写(集训后的自觉),但是发现如果使用这两种方法,反而会更加麻烦,效率上也不是很好,想了一下发下双指针还是比较好用的。
题目描述
传送门:https://leetcode.cn/problems/check-if-a-word-occurs-as-a-prefix-of-any-word-in-a-sentence/
简单的说:就是给你一个句子(字符串,带空格),一个前缀词,句子中有多个单词(>= 1),你需要找到前缀词是句子中单词的第一个单词。返回单词的下标(从1开始)。
示例
输入:sentence = "i love eating burger", searchWord = "burg"
输出:4
解释:"burg" 是 "burger" 的前缀,而 "burger" 是句子中第 4 个单词。
详细的可以进上面的传送门看这道题目。
提示
1 <= sentence.length <= 100
1 <= searchWord.length <= 10
sentence 由小写英文字母和空格组成。
searchWord 由小写英文字母组成。
解法——双指针
我们可以很容易想到暴力的解法,从句子中分离出单词,这里可以使用 split()
函数,将空格作为分隔符,即可实现将句子分离出单词,然后挨个匹配,当然也可以直接从句子中截取字符串匹配,这也行。
但是这两种方式,都或多或少需要往前回溯亦或者有额外的时间消耗。
而如果我们使用下面的双指针,即可实现只遍历一次句子,就可以得到答案。
在使用双指针前,我们需要思考一个重要的问题:指针的移动规则
。
指针的移动规则
对于前缀词的字母指针:
- 如果当前单词和前缀词的指针所指字母相同时,则指针向后移动;
- 如果不同时,需要重置为开头;
- 遇到空格时,也需要重置。
对于句子的指针:
- 一直移动;
- 如果当前单词和前缀词匹配失败,接下来的字母都跳过,直到下一个单词。
理清了移动规则,我们就可以快速实现代码了。
Code(C++)
#include<bits/stdc++.h>
typedef long long ll;
#define endl '\n'
typedef std::string string;
class Solution
{
public:
// 双指针
int isPrefixOfWord(string sentence, string searchWord)
{
bool flag = false; // 用于标记是否需要跳过
int n = sentence.length(), len = searchWord.length();
int ans = -1;
int ind = 1;
for(int i = 0, j = 0; i < n; ++i)
{
// 遇到空格的时候,需要将跳过标志重置,并将单词下标移动,并直接循环
if(sentence[i] == ' ')
{
flag = false;
ind++;
j = 0; // 防止前置的匹配但不完全的情况
continue;
}
// 跳过当前单词
if(flag)
{
continue;
}
// 当前单词的当前字母和检索词的字母相同
if(sentence[i] == searchWord[j])
{
++j; // 检索词字母指针移动
// 匹配成功
if(j == len)
{
ans = ind;
break;
}
}
// 匹配失败,需要重置检索词字母指针,并跳过接下来的字母单词,直到下一个单词
else
{
flag = true;
j = 0;
}
}
return ans;
}
};
int main()
{
string sen, sear;
Solution* sol = new Solution();
// 由于需要输入空格,所以采用按行读取
while(std::getline(std::cin, sen))
{
std::cin >> sear;
std::cout << sol->isPrefixOfWord(sen, sear) << endl;
getchar(); // 吞掉多余的换行
}
delete sol;
return 0;
}
通过截图(运行效率)
后话
加油吧!