数据结构学习:KMP算法

#include <iostream>
#include <string>
#define N 50
using namespace std;

/*主要思路是n[pos]=k+1等价于:*/
/*1.	0<=k<=j-1*/
/*2.	ptr[0]-ptr[k-1]每一位均与ptr[pos-k]-ptr[pos-1]相同*/
/*3.	ptr[k]=ptr[pos]*/
/*只要对满足1.、2.的k验证是否满足3.即可*/
/*得到特征向量的函数,采用了递归的思想*/
int GetN(char *ptr, int *n, int pos) {
	if (pos == 0) {			//第一位必为零,递归出口
		n[0] = 0;
		return 0;
	}
	else {
		int k = GetN(ptr, n, pos - 1);		//k即为特征向量的值,即如此得到的k一定满足条件1.、2.
	cmp:
		if (ptr[pos] == ptr[k]) {			//满足条件3.
			n[pos] = k + 1;					//等于k+1刚好是对应相同字符的个数
			return k + 1;
		}
		else if (k > 0) {					//从大到小,找满足1.、2.条件的前一个k值是否满足条件3.
			k = n[k - 1];
			goto cmp;
		}
		else {								//直到k=0也没找到则n[pos]=0
			n[pos] = 0;						
			return 0;
		}
	}
}

int main() {
	string str, ptr;		   //str为源字符串,ptr为目标字符串
	unsigned int i, j;		   //这里用unsigned是因为后来的'j<ptr.size()'会有warning......
	getline(cin, str);
	getline(cin, ptr);
	int n[N];				   //n为存放ptr各位置特征向量的数组
	GetN(&ptr[0], n, ptr.size() - 1);
	for (i = 0, j = 0; j < ptr.size();) {
		if (str.size() - i < ptr.size() - j) {		//若str[i]之后的字符个数少于ptr[j]之后的字符个数,则没有相同字符串
			cout << "not found" << endl;
			return 0;
		}
		else {
			if (ptr[j] != str[i]&&j != 0) {			//将ptr首位移到str[i-n[j-1]]的位置
				j = n[j - 1];
			}
			else if (ptr[j] != str[i] && j == 0) {		
				i++;
			}
			else if (ptr[j] == str[i])			
			{
				i++;
				j++;
			}
		}
	}
	cout << i - j;									//最终i-j的值即为所求位置
	return 0;
}

主要思路已经写在代码里了,个人感觉比较好的就是采用递归得到的k值一定满足条件1和条件2,只要对条件3进行判定就好了。

测试结果:

  

/*返回的是出现相同字符串的位置*/

/*如果各位发现了什么bug或者有什么建议请一定在评论区指出*/

### 回答1: 这道题目要求我们将字符串中每个单词的首字母变成大写。我们可以先将字符串按照空格分割成单词,然后对每个单词进行处理。处理的方法是将单词的第一个字符转换成大写,然后再将剩余的字符拼接起来。最后将处理后的单词再拼接成一个字符串即可。 ### 回答2: 题目要求编写一个程序,将给定字符串的每个单词的首字母变为大写。 首先,我们可以使用`split()`方法将字符串分割成单词的列表。然后,遍历每个单词,将首字母转化成大写。最后,使用`join()`方法将单词列表重新连接成字符串,得到结果。 以下是一个可能的实现: ```python def capitalize_first_letter(sentence): words = sentence.split() # 将字符串分割成单词列表 result = [] for word in words: capitalized_word = word[0].upper() + word[1:] # 将首字母变为大写 result.append(capitalized_word) return ' '.join(result) # 将单词列表重新连接成字符串 # 测试示例 print(capitalize_first_letter('hello world')) # 输出: Hello World print(capitalize_first_letter('zero starting point')) # 输出: Zero Starting Point ``` 以上代码中,`capitalize_first_letter`函接收一个字符串作为参,并返回首字母变为大写的结果字符串。我们将字符串使用`split()`方法分割成单词列表,然后遍历每个单词,将首字母转化成大写。最后,使用`join()`方法将单词列表重新连接成字符串,并返回结果。 通过上述实现,我们可以满足题目要求,将给定字符串的每个单词的首字母变为大写。 ### 回答3: 问题g:零起点算法106——首字母变大写 这个问题要求我们将给定字符串中每个单词的首字母变为大写。解决这个问题的方法有很多种,下面我将提供两种方法。 方法一:使用内置函capitalize() 我们可以使用内置函capitalize()来将字符串的首字母变为大写。首先,我们将给定的字符串按照空格分割成单词列表。然后,对于每个单词,我们使用capitalize()函将其首字母变为大写,并将其添加到结果列表中。最后,我们将结果列表连接成一个字符串,并返回这个结果字符串作为答案。 def capitalize_words(sentence): words = sentence.split() capitalized_words = [word.capitalize() for word in words] return ' '.join(capitalized_words) 方法二:逐个字符遍历 我们也可以逐个字符遍历给定字符串,当遇到空格时,将下一个字符变为大写。为了解决字符串首字母的问题,我们可以在遍历前在字符串的首部添加一个空格。 def capitalize_words(sentence): sentence = ' ' + sentence n = len(sentence) result = '' for i in range(1, n): if sentence[i - 1] == ' ': result += sentence[i].upper() else: result += sentence[i] return result 这两种方法都可以解决问题g中的要求,使用内置函capitalize()的方法较为简单和直观,但如果想要了解更多底层的实现细节,逐个字符遍历的方法也是一种不错的选择。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值