扩展KMP题目
hdu4333
/*
题意:字符串s[0..n-1],每次把最后一个字符放到前面,求形成的字符串比最初串分别小,相同,大于的个数
因为是为了练习扩展KMP所以肯定是扩展KMP,
为了循环方便,在后面复制字符串,求出next[]数组后,
如果next[i]>n那么肯定相等,如果小于就判断s[ next[i] ]和 s[ i+next[i] ]的大小判断 trick:题目求得是形成的不同的字符串的个数,可以知道相等的字符串肯定只有一个,
而从0..n-1,第二个next[i]大于n那么这个i就是该字符串的循环节;
*/
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<cmath>
#include<vector>
#include<algorithm>
using namespace std;
const int N=+;
char s[N*];
int next[N*],n; void getNext(char *s,int next[]){
int nn=strlen(s);
next[]=nn;
int p=;
while (s[p+] && s[p]==s[p+]) p++;
next[]=p;
int k=;
for (int i=;i<nn;i++){
int p=k+next[k]-, L=next[i-k];
if (i+L<=p) next[i]=L;
else{
int j=p-i+;
if (j<) j=;
while (i+j<nn && s[i+j]==s[j]) j++;
next[i]=j; k=i;
}
} } void work(){
int ret1,ret2,ret3;
ret1=ret2=ret3=;
getNext(s,next);
for (int i=;i<n;i++){
if (i!= && next[i]>=n) break;
if (next[i]<n) {
if (s[ i+next[i] ]<s[ next[i] ]) ret1++;
else ret3++;
}else ret2++;
}
printf("%d %d %d\n",ret1,ret2,ret3);
}
int main(){
int T,cas=;scanf("%d",&T);
while (T--){
scanf("%s",s);
printf("Case %d: ",++cas);
n=strlen(s);
for (int i=;i<n;i++){
s[n+i]=s[i];
}s[n+n]='\0';
// cout<<s<<endl;
work();
}
return ;
}
hdu3613
/*
题意:将一个串分成俩个串,如果是回文串值就是所有字符的值,如果不是值就为0
问你最大的值是多少; 分析:因为是扩展KMP的题,很容易想到用扩展KMP判读是否是回文串,而且题目的
特点是至少有一端是端点,这很容易想到前缀和后缀,然后思路就出来了;
首先将串s逆转变成s1,求出s的所有后缀与s1的最长公共子串的长度,
这样分割成两段的后面一段,判断该段是否是回文只要判断最长长度是否等于后面一段的 长度就可以了了
前面一段类似,只是要求出s1的所有后缀与s的最长公共子串的长度,判断方法类似 */
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#include<cmath>
#include<vector>
using namespace std;
const int N=+;
char s[N],s1[N];
int v[];
int ext1[N],ext2[N],next[N];
void getNext(char *s,int next[]){
int nn=strlen(s);
next[]=nn;
int p=;
while (s[p+] && s[p]==s[p+]) p++;
next[]=p;
int k=, L;
for (int i=;i<nn;i++){
p=k+next[k]-; L=next[i-k];
if (i+L<=p) next[i]=L;
else {
int j=p-i+;
if (j<) j=;
while (i+j<nn && s[i+j]==s[j]) j++;
next[i]=j; k=i;
}
} }
void getExtend(char *s,char *T,int *ext){
getNext(s,next);
int p=, nn=strlen(s);
while (s[p] && s[p]==T[p]) p++;
ext[]=p;
int k=, L;
for (int i=;i<nn;i++){
p=k+ext[k]-; L=next[i-k];
if (i+L<=p) ext[i]=L;
else {
int j=p-i+;
if(j<) j=;
while (i+j<nn && s[i+j]==T[j]) j++;
ext[i]=j; k=i;
}
}
}
int val[N];
void work(){
memset(val,,sizeof(val));
int len=strlen(s);
for (int i=;i<len;i++) val[i+]=val[i]+v[s[i]-'a']; for (int i=len-;i>=;i--) s1[len-i-]=s[i];
s1[len]='\0';
// cout<<s<<endl<<s1<<endl;
getExtend(s1,s,ext1);
getExtend(s,s1,ext2); int ret=;
for (int i=;i<len;i++){
int l1=i,l2=len-l1;
int tmp=;
if (ext1[len-i]==l1) tmp+=val[i];
if (ext2[i]==l2) tmp+=val[len]-val[i];
if (tmp>ret) ret=tmp;
}
printf("%d\n",ret);
}
int main(){
int T;scanf("%d",&T);
while (T--){
for (int i=;i<;i++) scanf("%d",v+i);
scanf("%s",s);
work();
}
return ;
}
/*
4
1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1
aaadbbbd
1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1
ab
1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1
aba
1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1
aaaaaaaaaaaaaaa */
poj1699
/*
题意:给你n个串,求最短的串使得n个串都包含在里面(n<=10)
从n的数据范围就可以看出状态压缩dp
dp[i][j]表示已经包含了"i"个串,且最后一个串是j的最小长度
预处理出s[i]的后缀与s[j]的前缀的最长公共子串 */
#include<cstdio>
#include<cstring>
#include<iostream>
#include<cmath>
#include<vector>
#include<algorithm>
using namespace std;
const int N=+;
char s[N][N];
int n,next[N],mz[N][N],ext[N];
void getNext(char *s,int *next){
int nn=strlen(s);
next[]=nn;
int p=;
while (s[p+] && s[p]==s[p+]) p++;
next[]=p;
int k=, L;
for (int i=;i<nn;i++){
p=k+next[k]-; L=next[i-k];
if (i+L<=p) next[i]=L;
else{
int j=p-i+;
if (j<) j=;
while (i+j<nn && s[i+j]==s[j]) j++;
next[i]=j; k=i;
}
}
}
void getExtend(char *s,char *T,int ext[]){
getNext(s,next);
int nn=strlen(s);
int p=;
while (s[p] && T[p] && s[p]==T[p]) p++;
ext[]=p;
int k=,L;
for (int i=;i<nn;i++){
p=k+ext[k]-; L=next[i-k];
if (i+L<=p) ext[i]=L;
else {
int j=p-i+;
if (j<) j=;
while (i+j<nn && s[i+j]==T[j]) j++;
ext[i]=j; k=i;
}
}
}
void init(){
memset(mz,,sizeof(mz));
for (int i=;i<n;i++){
for (int j=;j<n;j++){
if (i==j) continue;
getExtend(s[i],s[j],ext);
int nn=strlen(s[i]);
for (int k=;k<nn;k++){
if (ext[k]==nn-k) {
mz[i][j]=nn-k; break;
}
}
}
}
/*for (int i=0;i<n;i++){
for (int j=0;j<n;j++){
cout<<mz[i][j]<<" ";
}cout<<endl;
}*/
}
const int NN=<<;
int dp[NN][];
void Min(int &x,int y){
if (x==-) x=y;
else x=min(x,y);
}
void work(){
memset(dp,-,sizeof(dp));
for (int i=;i<n;i++){
dp[<<i][i]=strlen(s[i]);
}
for (int i=;i<(<<n);i++){
for (int j=;j<n;j++){
if (dp[i][j]==-) continue;
for (int k=;k<n;k++){
if ( i&(<<k) ) continue;
Min(dp[i^(<<k)][k],dp[i][j]+strlen(s[k])-mz[j][k]);
}
}
}
int ret=-;
for (int i=;i<n;i++){
Min(ret,dp[(<<n)-][i]);
}
cout<<ret<<endl; }
int main(){
int T;scanf("%d",&T);
while (T--){
scanf("%d",&n);
for (int i=;i<n;i++) scanf("%s",s[i]);
init();
work();
}
return ;
}
扩展KMP题目的更多相关文章
- hdu4333 扩展KMP
慢慢研究可以发现,可以用扩展kmp来求.由于扩展kmp的next[]只有一部分,当前位子前面那部分和母串的后部分,所以可以将字符串复制接在后面一次. 先求如果next[]>0&& ...
- HDU 3336 扩展kmp
题目大意: 找到字符串中所有和前缀字符串相同的子串的个数 对于这种前缀的问题,通常通过扩展kmp来解决 其实吧这是我第一次做扩展kmp的题目,原来确实看过这个概念,今天突然做到,所以这个扩展kmp的模 ...
- 【GDOI2014 DAY2】Beyond (扩展KMP)
[题目] [题意] Jodie和Aiden在做游戏.Jodie在一个长度为l字符串环上走路,他每离开一个就会记下格子当前字符.他让Aiden在他走了一圈后叫他停下来.Aiden决定耍一下Jodie,在 ...
- HDU 4333 Revolving Digits 扩展KMP
链接:http://acm.hdu.edu.cn/showproblem.php?pid=4333 题意:给以数字字符串,移动最后若干位到最前边,统计得到的数字有多少比原来大,有多少和原来同样,有多少 ...
- HDU 3613 Best Reward 正反两次扩展KMP
题目来源:HDU 3613 Best Reward 题意:每一个字母相应一个权值 将给你的字符串分成两部分 假设一部分是回文 这部分的值就是每一个字母的权值之和 求一种分法使得2部分的和最大 思路:考 ...
- (简单) POJ 1961 Period,扩展KMP。
Description For each prefix of a given string S with N characters (each character has an ASCII code ...
- A - A Secret -扩展KMP
题目大意: 给你两个字符串A,B,现在要你求B串的后缀在A串中出现的次数和后缀长度的乘积和为多少. 题解: 扩展KMP模板题,将A和B串都逆序以后就变成了求前缀的问题了,扩展KMP求处从i位置开始的最 ...
- HDU-6153 A Secret 扩展KMP
题意:求一个字符串的所有后缀在母串中的出现次数*后缀的长度的总和. 题目链接:http://acm.split.hdu.edu.cn/viewcode.php?rid=22147273 思路:先预处理 ...
- hdu 4300 Clairewd’s message(扩展kmp)
Problem Description Clairewd is a member of FBI. After several years concealing in BUPT, she interce ...
随机推荐
- Java中静态代码块,代码块,构造方法优先级、区别及代码示例
在项目中遇到了代码块的知识点,跑了下测试,写下结论 代码优先级:静态代码块 -> 构造代码块 -> 构造方法 多个代码块优先级,按照“先定义的代码先执行,后定义的代码后执行”原则执行 静态 ...
- 面试题_66_to_75_Java IO 和 NIO 的面试题
IO 是 Java 面试中一个非常重要的点.你应该很好掌握 Java IO,NIO,NIO2 以及与操作系统,磁盘 IO 相关的基础知识.下面是 Java IO 中经常问的问题. 66)在我 Java ...
- Android UI学习 - FrameLayou和布局优化(viewstub)
原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://android.blog.51cto.com/268543/308090 Fram ...
- IIS Server is too busy 解决方法(IIS6)
Server is too busy意思是服务器繁忙,资源不够用 为什么会出现这个问题呢? 因为服务器的配置不同,所能承受的压力不同. 而服务器默认对链接数,线程数等有设置,但这个设置太小,基本不够用 ...
- I.MX6 Android netperf
/***************************************************************************** * I.MX6 Android netpe ...
- noip2008提高组题解
第一题:笨小猴 模拟 第二题:火柴棒等式 搜索 深搜不用说,确定出两个加数然后判断能否拼出等式. 枚举确实不太好搞,因为枚举范围不确定,太大了容易超时,太小了容易漏解.不过这题的数据貌似很温和,我 ...
- ZOJ 1455 Schedule Problem(差分约束系统)
// 题目描述:一个项目被分成几个部分,每部分必须在连续的天数完成.也就是说,如果某部分需要3天才能完成,则必须花费连续的3天来完成它.对项目的这些部分工作中,有4种类型的约束:FAS, FAF, S ...
- jquery加入购物车飞入的效果
主要原理是:点击当前图片的时候,复制(克隆)当前图片在当前位置,然后利用jQuery的animate()方法实现图像的飞入效果 效果预览:http://runjs.cn/detail/qmf0mtm1 ...
- Solr部署如何启动
我刚接触solr,我要怎么启动,这是群里的朋友问得比较多的问题, solr最新版本下载地址: http://www.apache.org/dyn/closer.cgi/lucene/solr/ 1.准 ...
- Oracle 存储过程动态建表
动态sql,顾名思义就是动态执行的sql,也就是说在没执行之前是动态的拼接的. 任务 传入参数:新建的表名hd+当前的年和月,例如hd_201105表结构是:字段1:id ,类型是number,可以自 ...