B君的第二题



分析

题目就是让你求文本框与s匹配到了哪个位置。

考场70分

一看就是裸的kmp,直接打上去。

const int MAXN=1e5+8;
char s[MAXN],t[MAXN];
int n,m;
int nx[MAXN];
int f[MAXN],len; int main()
{
freopen("xining.in","r",stdin);
freopen("xining.out","w",stdout);
scanf("%s",s+1);
n=strlen(s+1);
scanf("%s",t+1);
m=strlen(t+1); for(int i=2;i<=n;++i)
{
int p=nx[i-1];
while(p&&s[p+1]!=s[i])
p=nx[p];
if(s[p+1]==s[i])
++p;
nx[i]=p;
} printf("%d\n",n);
for(int i=1;i<=m;++i)
{
if(t[i]=='-')
{
--len;
if(len<0)
len=0;
printf("%d\n",n-f[len]);
continue;
}
// cerr<<"pro "<<i<<endl;
int p=f[len];
// cerr<<" p="<<p<<endl;
while(p&&s[p+1]!=t[i])
p=nx[p];
if(s[p+1]==t[i])
++p;
f[++len]=p;
// cerr<<" p="<<p<<endl;
printf("%d\n",n-p);
}
// fclose(stdin);
// fclose(stdout);
return 0;
}

结果TLE了。实际上时间复杂度\(O(n^2)\)

标解

这道题能卡kmp,是因为往回走后重新匹配跳next时复杂度没了保证。

退格操作可以让匹配长度剧增,这与kmp要满足长度增加量最多为文本串的长度的性质不符,所以要被卡。

B君:可以前n/2都是a,然后后面一个-一个b交替出现,就可以把kmp卡成n方的了。

但是AC自动机中的一个构建优化是把不存在的儿子补齐,所以考虑AC自动机。

然后因为AC自动机是在Trie图上建的,距离不能直接求,所以要预处理。

时间复杂度\(O(n)\)

const int MAXN=1e5+7;

vector<int>a[MAXN];
int d[MAXN]; const int CHARSET=26;
struct Trie
{
int sz;
int to[MAXN][CHARSET];
int nx[MAXN];
int fa[MAXN];
int v[MAXN]; il void ins(rg char*s)
{
rg int p=0;
for(;*s;++s)
{
rg int c=*s-'a';
if(to[p][c]==0)
{
to[p][c]=++sz;
fa[sz]=p;
}
p=to[p][c];
}
v[p]=1;
} il void build()
{
rg queue<int>Q;
Q.push(0);
while(!Q.empty())
{
rg int x=Q.front();
Q.pop();
v[x]|=v[nx[x]];
for(rg int i=0;i<CHARSET;++i)
{
if(to[x][i])
{
nx[to[x][i]]=x?to[nx[x]][i]:0;
Q.push(to[x][i]);
}
else
{
to[x][i]=x?to[nx[x]][i]:0;
}
}
}
for(rg int i=0;i<=sz;++i)
for(rg int j=0;j<CHARSET;++j)
{
a[to[i][j]].push_back(i);
}
memset(d,0x3f,sizeof d);
for(rg int i=0;i<=sz;++i)
if(v[i])
{
Q.push(i);
d[i]=0;
}
while(!Q.empty())
{
rg int x=Q.front();
Q.pop();
for(rg int i=0;i<a[x].size();++i)
{
rg int y=a[x][i];
if(d[y]>d[x]+1)
{
d[y]=d[x]+1;
Q.push(y);
}
}
}
}
}T; char s[MAXN],t[MAXN]; int f[MAXN],len; int main()
{
freopen("xining.in","r",stdin);
freopen("xining.out","w",stdout);
scanf("%s",s);
T.ins(s);
T.build();
scanf("%s",t);
printf("%d\n",d[f[len]]);
for(rg int i=0;t[i];++i)
{
if(t[i]=='-')
{
if(len>0)
--len;
}
else
{
rg int x=T.to[f[len]][t[i]-'a']; // edit 1
f[++len]=x;
}
// cerr<<"f="<<f[len]<<endl;
printf("%d\n",d[f[len]]);
}
// fclose(stdin);
// fclose(stdout);
return 0;
}

test20181020 B君的第二题的更多相关文章

  1. B 君的第二题 (hongkong)

    B 君的第二题 (hongkong) 题目大意: 一个长度为\(n(n\le2\times10^5)\)的数组,给定一个数\(k(k\le40)\).用\(a[i][j]\)表示该数组\(i\)次前缀 ...

  2. test20181016 B君的第二题

    题意 分析 考场暴力50分. 考虑bfs序,一个点的儿子节点的bfs序一定连续,所以对bfs序建线段树,努力打一下就行了. 时间复杂度\(O(n \log n + m \log n)\) #inclu ...

  3. test20181017 B君的第二题

    题意 分析 考场50分 旁边的L君告诉我,求的就是非升子序列的个数,于是写了个树状数组. 但是\(\mod{2333} > 0\)还需要组合数中没有2333的倍数,所以实际上只得了\(a_i \ ...

  4. test20181015 B君的第二题

    题意 分析 考场85分 用multiset暴力,由于教练的机子飞快,有写priority_queue水过了的人. #include<cstdlib> #include<cstdio& ...

  5. test20181020 B君的第一题

    题意 分析 二次剩余问题. x,y相当于二次方程 \[ x^2-bx+c=0 \mod{p} \] 的两根. 摸意义下的二次方程仍然考虑判别式\(\Delta=b^2-4c\). 它能开根的条件是\( ...

  6. test20181019 B君的第二题

    题意 分析 快速子集和变换以及快速超集和变换的裸题. 用\(f(s)\)表示集合s的方案数,初始化为输入中s出现的次数. 做一遍快速子集和变换,此时f(s)表示s及其子集在输入中出现的次数. 对所有f ...

  7. Java蓝桥杯02——第二题集锦:生日蜡烛、星期一、方格计数、猴子分香蕉

    第二题 生日蜡烛(结果填空) 某君从某年开始每年都举办一次生日party,并且每次都要吹熄与年龄相同根数的蜡烛. 现在算起来,他一共吹熄了236根蜡烛. 请问,他从多少岁开始过生日party的? 请填 ...

  8. 05:统计单词数【NOIP2011复赛普及组第二题】

    05:统计单词数 总时间限制:  1000ms 内存限制:  65536kB 描述 一般的文本编辑器都有查找单词的功能,该功能可以快速定位特定单词在文章中的位置,有的还能统计出特定单词在文章中出现的次 ...

  9. 常见面试第二题之什么是Context

    今天的面试题,也就是我们常见面试题系列的第二题,我们来讲一讲android中的context.我相信大家android开发者一定对于这个context非常熟悉,肯定都有使用过,肯定没有没使用过的.但是 ...

随机推荐

  1. uva1401 dp+Trie

    这题说的是给了一个长的字符串长度最大300000,又给了4000个单词 单词的长度不超过100.计算这个字符串能组成多少种不同单词的组合,求出方案总数.dp[i]以第i个字符为开始的字符串能有多少种的 ...

  2. SQL :模糊查询,转义字符

    1. 查询table表name列包含 '_BCE' 的记录 select * from table where name like '_BCE%' ABCEDF _BCEFG _BCEDF 3 row ...

  3. 清理tomcat日志大的文件

    先看一个命令: [root@weblogic logs]# catalina.--.log icatalina.--:-.out icatalina.--:-.out localhost_access ...

  4. ZLYD团队第三周项目总结

    ZLYD团队第三周项目总结 项目进展 我们的吃豆子游戏的程序由八个文件组成:Wall.java.Gold.java.Player.java.Fruit.java.Enemy.java.Ticker.j ...

  5. CodeForces 828C String Reconstruction(并查集思想)

    题意:给你n个串,给你每个串在总串中开始的每个位置,问你最小字典序总串. 思路:显然这道题有很多重复填涂的地方,那么这里的时间花费就会特别高. 我们维护一个并查集fa,用fa[i]记录从第i位置开始第 ...

  6. Python学习札记(三十六) 面向对象编程 Object Oriented Program 7 __slots__

    参考:slots NOTE 1.动态语言灵活绑定属性及方法. #!/usr/bin/env python3 class MyClass(object): def __init__(self): pas ...

  7. java 中 Cannot make a static reference to the non-static 解决方法

    今天敲代码的时候遇到了这个问题,大体这个问题可以简化成这样: public class Test1 { public String get() { return "123"; } ...

  8. Linux的硬链接和软链接

    1.Linux链接概念Linux链接分两种,一种被称为硬链接(Hard Link),另一种被称为符号链接(Symbolic Link), 也就是软链接.默认情况下,ln命令产生硬链接. [硬连接]硬连 ...

  9. Uva 11178 Morley's Theorem 向量旋转+求直线交点

    http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=9 题意: Morlery定理是这样的:作三角形ABC每个 ...

  10. 这些HTML、CSS知识点,面试和平时开发都需要 No1-No4(知识点:HTML、CSS、盒子模型、内容布局)

    这些HTML.CSS知识点,面试和平时开发都需要 No1-No4   系列知识点汇总 这些HTML.CSS知识点,面试和平时开发都需要 No1-No4(知识点:HTML.CSS.盒子模型.内容布局) ...