1455 检查单词是否为句中其他单词的前缀——Leetcode天天刷(2022.8.21)【双指针】

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() 函数,将空格作为分隔符,即可实现将句子分离出单词,然后挨个匹配,当然也可以直接从句子中截取字符串匹配,这也行。

但是这两种方式,都或多或少需要往前回溯亦或者有额外的时间消耗。

而如果我们使用下面的双指针,即可实现只遍历一次句子,就可以得到答案。

在使用双指针前,我们需要思考一个重要的问题:指针的移动规则

指针的移动规则

对于前缀词的字母指针:

  1. 如果当前单词和前缀词的指针所指字母相同时,则指针向后移动;
  2. 如果不同时,需要重置为开头;
  3. 遇到空格时,也需要重置。

对于句子的指针:

  1. 一直移动;
  2. 如果当前单词和前缀词匹配失败,接下来的字母都跳过,直到下一个单词。

理清了移动规则,我们就可以快速实现代码了。

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;
}

通过截图(运行效率)

在这里插入图片描述

后话

加油吧!

评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值