HDU3336Count the string(dp+kmp_next表性质)

题目链接hdu3336

题目大意:给一串n长度的字符串str,问str所有前缀(包括本串)在str中的匹配次数。

解题思路:利用kmp爆搜得到一发TLE,参考kmp+dp 以及dp状态方程解释博客思路学会利用next数组的性质进行dp优化:next[i]表示不包含下标为i的字符(i下标之前的串)的最长前后缀相等长度,那么我们设dp[i]表示以第i个字符为结尾 (或者说长度为i)的前缀能够匹配到的前缀个数,思考一下:如果我们得到长度为i的前缀串,那么本身一定匹配1次,然后前面i-1长度的最长前后缀匹配1次(这里i-1的最长前后缀匹配次数要通过dp[next[i]]状态转移过来,因为next[i]的性质,然后通过这个长度取到匹配到的个数),最后两部分加起来就是dp[i]即长度为i的前缀匹配到的前缀个数,这样我们就能很容易得到 转移方程 : dp[i] = dp[next[i]] + 1


kmp+爆搜超时代码(思路局限到模板模拟导致)

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>
#include <iostream>
#include <algorithm>
#include <string>
#include <deque>
#include <queue>
#include <set>
#include <map>
#include <stack>
#include <vector>
#include <utility>

using namespace std;
const int maxn = (int)2e5+5;
const int mod = 10007;

int prefix[maxn];

void get_prefix_table (string& pattern,int n) {
	int i = 0,len = -1;
	prefix[0] = -1;
	while (i < n) {
		if(len == -1 || pattern.at(i) == pattern.at(len)) {
			i++;
			len++;
			prefix[i] = len;
		}else {
			len = prefix[len];
		}
	}
	return ;
}

int kmp_search (string& tmp, string& pattern) {
	int i = 0,j = 0,n,m,ans = 0;
	n = tmp.size();
	m = pattern.size();
	get_prefix_table(pattern, m);
	while (i < n) {
		if(j == -1 || tmp.at(i) == tmp.at(j)) {
			i++;
			j++;
		}else {
			j = prefix[j];
		}
		if(j == m) {
			ans++;
			j = prefix[j];
		}
	}
	return ans;
}

int solve	(string& tmp) {
	int ans = 0;
	string pattern;
	for(int i = 1; i <= tmp.size(); i++) {
		pattern = tmp.substr(0,i);
		ans = (ans + kmp_search(tmp, pattern))%mod;
	}
	return ans;
}

int main() {
	ios::sync_with_stdio(false);
	int T,n;
	string str;
	cin >> T;
	while (T--) {
		cin >> n >> str;
		cout << solve(str) << '\n';
	}
	return 0;
}

AC代码

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>
#include <iostream>
#include <algorithm>
#include <string>
#include <deque>
#include <queue>
#include <set>
#include <map>
#include <stack>
#include <vector>
#include <utility>

using namespace std;
const int maxn = (int)2e5+5;
const int mod = 10007;

int prefix[maxn],dp[maxn];
string pattern;

void get_prefix_table() {
	int i = 0,len = -1,n = pattern.size();
	prefix[0] = -1;
	while (i < n) {
		if(len == -1 || pattern.at(i) == pattern.at(len)) {
			i++;
			len++;
			prefix[i] = len;
		}else {
			len = prefix[len];
		}
	}
	return ;
}

int main() {
	ios::sync_with_stdio(false);
	int T,n,sum;
	cin >> T;
	while (T--) {
		cin >> n >> pattern;
		get_prefix_table();
		memset(dp,0,sizeof(dp));
		dp[0] = 0;
		sum = 0;
		for (int i = 1; i <= n; i++) {
			dp[i] = (dp[prefix[i]] + 1) % mod;
			sum = (sum + dp[i]) % mod;
		}
		cout << sum << '\n';
	}
	return 0;
}
全部评论

相关推荐

06-26 17:24
已编辑
宁波大学 golang
迷失西雅图:别给,纯kpi,别问我为什么知道
点赞 评论 收藏
分享
06-27 12:54
已编辑
门头沟学院 Java
累了,讲讲我的大学经历吧,目前在家待业。我是一个二本院校软件工程专业。最开始选专业是觉得计算机感兴趣,所以选择了他。本人学习计算机是从大二暑假结束开始的,也就是大三开始。当时每天学习,我个人认为Java以及是我生活的一部分了,就这样持续学习了一年半,来到了大四上学期末,大概是在12月中旬,我终于找的到了一家上海中厂的实习,但我发现实习生的工作很枯燥,公司分配的活也不多,大多时间也是自己在自学。就这样我秋招末才找到实习。时间来到了3月中旬,公司说我可以转正,但是转正工资只有7000,不过很稳定,不加班,双休,因为要回学校参加答辩了,同时当时也是心高气傲,认为可以找到更好的,所以放弃了转正机会,回学校准备论文。准备论文期间就也没有投递简历。然后时间来到了5月中旬,这时春招基本也结束了,然后我开始投递简历,期间只是约到了几家下场面试。工资也只有6-7k,到现在我不知道该怎么办了。已经没有当初学习的心劲了,好累呀,但是又不知道该干什么去。在家就是打游戏,boss简历投一投。每天日重一次。26秋招都说是针对26届的人,25怎么办。我好绝望。要不要参加考公、考研、央国企这些的。有没有大佬可以帮帮我。为什么感觉别人找工作都是顺其自然的事情,我感觉自己每一步都在艰难追赶。八股文背了又忘背了又忘,我每次都花很长时间去理解他,可是现在感觉八股、项目都忘完了。真的已经没有力气再去学习了。图片是我的简历,有没有大哥可以指正一下,或者说我应该走哪条路,有点不想在找工作了。
码客明:太累了就休息一下兄弟,人生不会完蛋的
如果实习可以转正,你会不...
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务